一、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執行");
}
}

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/180055.html
標籤:Java
上一篇:SOFA入門
