父類引用指向子類

Java 轉型問題其實並不複雜,只要記住一句話:父類引用指向子類對象。

什麼叫父類引用指向子類對象,且聽我慢慢道來。

從 2 個名詞開始說起:向上轉型(upcasting) 、向下轉型(downcasting)

舉個例子:有2個類,Father 是父類,Son 類繼承自 Father。

第 1 個例子:

Father f1 = new Son();   // 這就叫 upcasting (向上轉型)
// 現在 f1 引用指向一個Son對象

Son s1 = (Son)f1;   // 這就叫 downcasting (向下轉型)
// 現在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 引用不能指向父類對象。

總結:

1、父類引用指向子類對象,而子類引用不能指向父類對象。

2、把子類對象直接賦給父類引用叫upcasting向上轉型,向上轉型不用強制轉換嗎,如:

Father f1 = new Son();

3、把指向子類對象的父類引用賦給子類引用叫向下轉型(downcasting),要強制轉換,如:

f1 就是一個指向子類對象的父類引用。把f1賦給子類引用 s1 即 Son s1 = (Son)f1;

其中 f1 前面的(Son)必須加上,進行強制轉換。

一、向上轉型。

通俗地講即是將子類對象轉爲父類對象。此處父類對象可以是接口。

1、向上轉型中的方法調用:

實例

public class Animal { public void eat(){ System.out.println("animal eatting..."); } } class Bird extends Animal{ public void eat(){ System.out.println("bird eatting..."); } public void fly(){ System.out.println("bird flying..."); } } class Main{ public static void main(String[] args) { Animal b=new Bird(); //向上轉型 b.eat(); //! error: b.fly(); b雖指向子類對象,但此時丟失fly()方法 dosleep(new Male()); dosleep(new Female()); } public static void dosleep(Human h) { h.sleep(); } }

實例

public class Human { public void sleep() { System.out.println("Human sleep.."); } } class Male extends Human { @Override public void sleep() { System.out.println("Male sleep.."); } } class Female extends Human { @Override public void sleep() { System.out.println("Female sleep.."); } }

注意這裏的向上轉型:

Animal b=new Bird(); //向上轉型
b.eat();

此處將調用子類的 eat() 方法。原因:b 實際指向的是 Bird 子類,故調用時會調用子類本身的方法。

需要注意的是向上轉型時 b 會遺失除與父類對象共有的其他方法。如本例中的 fly 方法不再爲 b 所有。

2、向上轉型的好處

看上面的代碼:

public static void dosleep(Human h) {
    h.sleep();
}

這裏以父類爲參數,調有時用子類作爲參數,就是利用了向上轉型。這樣使代碼變得簡潔。不然的話,如果 dosleep 以子類對象爲參數,則有多少個子類就需要寫多少個函數。這也體現了 JAVA 的抽象編程思想。

二、向下轉型。

與向上轉型相反,即是把父類對象轉爲子類對象。

實例

package com.wensefu.other1; public class Girl { public void smile(){ System.out.println("girl smile()..."); } } class MMGirl extends Girl{ @Override public void smile() { System.out.println("MMirl smile sounds sweet..."); } public void c(){ System.out.println("MMirl c()..."); } } class Main{ public static void main(String[] args) { Girl g1=new MMGirl(); //向上轉型 g1.smile(); MMGirl mmg=(MMGirl)g1; //向下轉型,編譯和運行皆不會出錯 mmg.smile(); mmg.c(); Girl g2=new Girl(); // MMGirl mmg1=(MMGirl)g2; //不安全的向下轉型,編譯無錯但會運行會出錯 // mmg1.smile(); // mmg1.c(); /*output: * CGirl smile sounds sweet... * CGirl smile sounds sweet... * CGirl c()... * Exception in thread "main" java.lang.ClassCastException: com.wensefu.other1.Girl * at com.wensefu.other1.Main.main(Girl.java:36) */ if(g2 instanceof MMGirl){ MMGirl mmg1=(MMGirl)g2; mmg1.smile(); mmg1.c(); } } }

Girl g1=new MMGirl(); //向上轉型
g1.smile();
MMGirl mmg=(MMGirl)g1; //向下轉型,編譯和運行皆不會出錯

這裏的向下轉型是安全的。因爲 g1 指向的是子類對象。

Girl g2=new Girl();
MMGirl mmg1=(MMGirl)g2; //不安全的向下轉型,編譯無錯但會運行會出錯

運行出錯:

Exception in thread "main" java.lang.ClassCastException: com.wensefu.other1.Girl
    at com.wensefu.other1.Main.main(Girl.java:36)

如代碼所示,可以通過 instanceof 來防止出現異常。

發佈了19 篇原創文章 · 獲贊 2 · 訪問量 4241
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章