重寫:首頁必須有繼承關係,子類重寫父類方法即是將父類的方法重新定義,重寫的規則:除了訪問修飾符和方法主體其餘的部分必須相同(但是子類的訪問修飾符必須比父類的更加嚴格)。
方法重寫是多態的體現。
重載:主要是在同一類中方法名相同,參數列表不同的方法。
參數列表不同主要體現在:
1.參數個數不同
2.參數類型不同
3.在參數類型不同的基礎上參數次序不同
例:
public void run(int a,double b){
}
public void run(int a){
}
public void run(double b,int a){
}
例:-------方法重載----------
public void run(int a,double b){
}
public void run(int a){
}
public void run(double b,int a){
}
----------方法重寫:------
public class Animal{
public void run(){
System.out.print("animal run ....");
}
}
public class dog extends Animal{ //繼承Animal類
public void run(){ //重寫父類的run方法
System.out.print("dog run ....");
}
}
(1)方法重載是讓類以統一的方式處理不同類型數據的一種手段。多個同名函數同時存在,具有不同的參數個數/類型。重載Overloading是一個類中多態性的一種表現。
(2)Java的方法重載,就是在類中可以創建多個方法,它們具有相同的名字,但具有不同的參數和不同的定義。調用方法時通過傳遞給它們的不同參數個數和參數類型來決定具體使用哪個方法, 這就是多態性。
(3)重載的時候,方法名要一樣,但是參數類型和個數不一樣,返回值類型可以相同也可以不相同。無法以返回型別作爲重載函數的區分標準。
下面是重載的例子:
package c04.answer;//這是包名
//這是這個程序的第一種編程方法,在main方法中先創建一個Dog類實例,然後在Dog類的構造方法中利用this關鍵字調用不同的bark方法。不同的重載方法bark是根據其參數類型的不同而區分的。
//注意:除構造器以外,編譯器禁止在其他任何地方中調用構造器。
package c04.answer;
public class Dog {
Dog()
{
this.bark();
}
void bark()//bark()方法是重載方法
{
System.out.println("nobarking!");
this.bark("female",3.4);
}
void bark(String m,double l)//注意:重載的方法的返回值都是一樣的,
{
System.out.println("abarking dog!");
this.bark(5,"China");
}
void bark(int a,String n)//不能以返回值區分重載方法,而只能以“參數類型”和“類名”來區分
{
System.out.println("ahowling dog");
}
public static void main(String[]args)
{
Dog dog = new Dog();
//dog.bark();
//dog.bark("male", "yellow");
//dog.bark(5,"China");
然後我們再來談談重寫(Overriding)
(1) 父類與子類之間的多態性,對父類的函數進行重新定義。如果在子類中定義某方法與其父類有相同的名稱和參數,我們說該方法被重寫 (Overriding)。在Java中,子類可繼承父類中的方法,而不需要重新編寫相同的方法。但有時子類並不想原封不動地繼承父類的方法,而是想作一定的修改,這就需要採用方法的重寫。方法重寫又稱方法覆蓋。
(2) 若子類中的方法與父類中的某一方法具有相同的方法名、返回類型和參數表,則新方法將覆蓋原有的方法。如需父類中原有的方法,可使用super關鍵字,該關鍵字引用了當前類的父類。
(3) 子類函數的訪問修飾權限不能少於父類的;下面是重寫的例子:
概念:即調用對象方法的機制。
動態綁定的內幕:
1、編譯器檢查對象聲明的類型和方法名,從而獲取所有候選方法。試着把上例Base類的test註釋掉,這時再編譯就無法通過。
2、重載決策:編譯器檢查方法調用的參數類型,從上述候選方法選出唯一的那一個(其間會有隱含類型轉化)。如果編譯器找到多於一個或者沒找到,此時編譯器就會報錯。試着把上例Base類的test(byte b)註釋掉,這時運行結果是1 1。
3、若方法類型爲priavte static final ,java採用靜態編譯,編譯器會準確知道該調用哪個方法。
4、當程序運行並且使用動態綁定來調用一個方法時,那麼虛擬機必須調用對象的實際類型相匹配的方法版本。在例子中,b所指向的實際類型是TestOverriding,所以b.test(0)調用子類的test。但是,子類並沒有重寫test(byte b),所以b.test((byte)0)調用的是父類的test(byte b)。如果把父類的(byte b)註釋掉,則通過第二步隱含類型轉化爲int,最終調用的是子類的test(int i)。
學習總結:多態性是面向對象編程的一種特性,和方法無關。
簡單說,就是同樣的一個方法能夠根據輸入數據的不同,做出不同的處理,即方法的重載——有不同的參數列表(靜態多態性)
而當子類繼承自父類的相同方法,輸入數據一樣,但要做出有別於父類的響應時,你就要覆蓋父類方法,即在子類中重寫該方法——相同參數,不同實現(動態多態性)
(2)
(3)
(4)
(5)
(6)OOP三大特性:繼承,多態,封裝。
public class Base
{
void test(int i)
{
System.out.print(i);
}
void test(byte b)
{
System.out.print(b);
}
}
public class TestOverriding extends Base
{
void test(int i)
{
i++;
System.out.println(i);
}
public static voidmain(String[]agrs)
{
Base b=new TestOverriding();
b.test(0)
b.test((byte)0)
}
}
這時的輸出結果是1 0,這是運行時動態綁定的結果。動態綁定