Java 轉(zhuǎn)型問題其實并不復(fù)雜,只要記住一句話:父類引用指向子類對象。
什么叫父類引用指向子類對象,且聽我慢慢道來。
從 2 個名詞開始說起:向上轉(zhuǎn)型(upcasting) 、向下轉(zhuǎn)型(downcasting)。
舉個例子:有2個類,F(xiàn)ather 是父類,Son 類繼承自 Father。
第 1 個例子:
Father f1 = new Son(); // 這就叫 upcasting (向上轉(zhuǎn)型) // 現(xiàn)在 f1 引用指向一個Son對象 Son s1 = (Son)f1; // 這就叫 downcasting (向下轉(zhuǎn)型) // 現(xiàn)在f1 還是指向 Son對象
第 2 個例子:
Father f2 = new Father(); Son s2 = (Son)f2; // 出錯,子類引用不能指向父類對象
你或許會問,第1個例子中:Son s1 = (Son)f1; 問為什么是正確的呢。
很簡單因為 f1 指向一個子類對象,Father f1 = new Son(); 子類 s1 引用當然可以指向子類對象了。
而 f2 被傳給了一個 Father 對象,Father f2 = new Father(); 子類 s2 引用不能指向父類對象。
總結(jié):
1、父類引用指向子類對象,而子類引用不能指向父類對象。
2、把子類對象直接賦給父類引用叫upcasting向上轉(zhuǎn)型,向上轉(zhuǎn)型不用強制轉(zhuǎn)換嗎,如:
Father f1 = new Son();
3、把指向子類對象的父類引用賦給子類引用叫向下轉(zhuǎn)型(downcasting),要強制轉(zhuǎn)換,如:
f1 就是一個指向子類對象的父類引用。把f1賦給子類引用 s1 即 Son s1 = (Son)f1;
其中 f1 前面的(Son)必須加上,進行強制轉(zhuǎn)換。
一、向上轉(zhuǎn)型。
通俗地講即是將子類對象轉(zhuǎn)為父類對象。此處父類對象可以是接口。
1、向上轉(zhuǎn)型中的方法調(diào)用:
實例
實例
注意這里的向上轉(zhuǎn)型:
Animal b=new Bird(); //向上轉(zhuǎn)型 b.eat();
此處將調(diào)用子類的 eat() 方法。原因:b 實際指向的是 Bird 子類,故調(diào)用時會調(diào)用子類本身的方法。
需要注意的是向上轉(zhuǎn)型時 b 會遺失除與父類對象共有的其他方法。如本例中的 fly 方法不再為 b 所有。
2、向上轉(zhuǎn)型的好處
看上面的代碼:
public static void dosleep(Human h) { h.sleep(); }
這里以父類為參數(shù),調(diào)有時用子類作為參數(shù),就是利用了向上轉(zhuǎn)型。這樣使代碼變得簡潔。不然的話,如果 dosleep 以子類對象為參數(shù),則有多少個子類就需要寫多少個函數(shù)。這也體現(xiàn)了 JAVA 的抽象編程思想。
二、向下轉(zhuǎn)型。
與向上轉(zhuǎn)型相反,即是把父類對象轉(zhuǎn)為子類對象。
實例
Girl g1=new MMGirl(); //向上轉(zhuǎn)型 g1.smile(); MMGirl mmg=(MMGirl)g1; //向下轉(zhuǎn)型,編譯和運行皆不會出錯
這里的向下轉(zhuǎn)型是安全的。因為 g1 指向的是子類對象。
而
Girl g2=new Girl(); MMGirl mmg1=(MMGirl)g2; //不安全的向下轉(zhuǎn)型,編譯無錯但會運行會出錯
運行出錯:
Exception in thread "main" java.lang.ClassCastException: com.wensefu.other1.Girl at com.wensefu.other1.Main.main(Girl.java:36)
如代碼所示,可以通過 instanceof 來防止出現(xiàn)異常。