一、java覆蓋如何執行:編譯看左邊,運行看右邊 在子類方法覆蓋父類方法時,在編譯期,編譯器會檢查這個對象的引用類型是否含有此方法。如果沒有則編譯會出錯,有則會通過編譯。但在執行期,JVM尋找的不是引用指向的類型,而是堆上的對象。(即編譯看左邊,運行看右邊)。 1.子類方法覆蓋了父類方法 //Fat ...
一、java覆蓋如何執行:編譯看左邊,運行看右邊
在子類方法覆蓋父類方法時,在編譯期,編譯器會檢查這個對象的引用類型是否含有此方法。如果沒有則編譯會出錯,有則會通過編譯。但在執行期,JVM尋找的不是引用指向的類型,而是堆上的對象。(即編譯看左邊,運行看右邊)。
1.子類方法覆蓋了父類方法
//Father類
public class Father {
public void turnOn(){
System.out.println("父類方法turnOn執行");
}
}
//Son類,繼承了Father類
public class Son extends Father {
@Override
public void turnOn() {
System.out.println("子類方法turnOn執行");
}
}
public class Demo {
public static void main(String[] args) {
Father f = new Son();
f.turnOn();
}
}
運行Demo類中的main()方法:
在編譯期,編譯器會檢查對象f的引用類型(即Father)是否有turnOn(),由於Father類有這個方法,於是編譯器不會報錯,程式編譯成功。在運行期,究竟運行Father類的兩個方法還是Son類的方法呢?JVM會從對象f開始向上找直到找到turnOn()為止,由於我們在Son類覆蓋了turnOn(),所以JVM會執行Son類的turnOn()。
2.父類引用不能調用子類特有的方法
//將上面的Son.class修改成下麵這個
public class Son extends Father {
@Override
public void turnOn() {
System.out.println("子類方法turnOn執行");
}
//定義一個只有子類有父類沒有的方法
public void sonFunction(){
System.out.println("子類獨有方法");
}
}
修改Demo的main()為下圖的代碼,編譯,可以看到編譯器找不到sonFunction()。為什麼呢?因為根據覆蓋的原理,f的引用類型(Father)是沒有sonFunction()方法的,所以就會編譯報錯。
3.父類引用調用子類特有的方法?利用instanceof或抽象類
下麵只介紹instanceof,抽象類快忘光了,下次再補充...(反正應該沒人看我的文章)。
將Demo的代碼修改成下圖:
public class Demo {
public static void main(String[] args) {
Father f = new Son();
f.turnOn();
if(f instanceof Son){
//對象A instanceof B,如果對象A是類B的對象,則返回true,否則返回false
Son s = (Son) f;//向下轉型,將f由Father類的引用轉為Son類的引用
s.sonFunction();//調用Son類獨有的方法
}
}
}
二、java子類方法覆蓋父類方法應該遵守的規則
1、參數必須要一樣,且返回類型要相容
父類方法使用了哪種參數,覆蓋此方法的子類也一定要使用相同的參數。不論父類的聲明的返回類型是什麼,子類必須聲明返回一樣的類型或該類型的子類型。
2、不能降低方法的存取許可權
這代表存取許可權必須相同,或者更為開放。舉例來說,你不能在子類中覆蓋一個公有方法並將它標記為私有。這會讓JVM在編譯期以為通過的是個公有方法,然後在執行期才會被JVM阻止存取。
public class Father {
public void turnOn(){
System.out.println("父類方法turnOn執行");
}
}
public class Son extends Father {
@Override
private void turnOn() {//此處報錯!!!!報錯信息如下圖
System.out.println("子類方法turnOn執行");
}
}