Java向上轉(zhuǎn)型和向下轉(zhuǎn)型實(shí)例解析
這篇文章主要介紹了Java向上轉(zhuǎn)型和向下轉(zhuǎn)型實(shí)例解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
學(xué)習(xí)向上轉(zhuǎn)型和向下轉(zhuǎn)型怎么用沒多難,但是為什么那樣用,我搞了很多次沒弄明白。沒弄明白的原因是平時(shí)學(xué)習(xí)時(shí)之看例子,而例子一般都比較簡單,沒有對象之間的調(diào)用,一般就是一個(gè)對象調(diào)用自己的方法。
首先看下怎么用轉(zhuǎn)型。
要轉(zhuǎn)型,首先要有繼承。繼承是面向?qū)ο笳Z言中一個(gè)代碼復(fù)用的機(jī)制,簡單說就是子類繼承了父類中的非私有屬性和可以繼承的方法,然后子類可以繼續(xù)擴(kuò)展自己的屬性及方法。
向上轉(zhuǎn)型:子類對象轉(zhuǎn)為父類,父類可以是接口。公式:Father f = new Son();Father是父類或接口,son是子類。
向下轉(zhuǎn)型:父類對象轉(zhuǎn)為子類。公式:Son s = (Son)f;
例子:向上轉(zhuǎn)型
package multistate; public class Human { public void sleep() { System.out.println("Human sleep.."); } public static void main(String[] args) { Human h = new Male();// 向上轉(zhuǎn)型 h.sleep(); Male m = new Male();// 干嘛要向上轉(zhuǎn)型 m.sleep(); // h.speak();此方法不能編譯,報(bào)錯(cuò)說Human類沒有此方法 } } class Male extends Human { @Override public void sleep() { System.out.println("Male sleep.."); } public void speak() { System.out.println("I am Male"); } } class Female extends Human { @Override public void sleep() { System.out.println("Female sleep.."); } public void speak() { System.out.println("I am Female"); } }
注意:向上轉(zhuǎn)型不要強(qiáng)制轉(zhuǎn)型。向上轉(zhuǎn)型后父類的引用所指向的屬性是父類的屬性,如果子類重寫了父類的方法,那么父類引用指向的或者調(diào)用的方法是子類的方法,這個(gè)叫動態(tài)綁定。向上轉(zhuǎn)型后父類引用不能調(diào)用子類自己的方法,就是父類沒有但是子類的方法,如果調(diào)用不能編譯通過,比如子類的speak方法。
非要調(diào)用子類的屬性呢?如果不向下轉(zhuǎn)型就需要給需要的屬性寫getter方法。
例子:
package multistate; public class Human { String name = "Human"; public String getName(){ return this.name; } public void sleep() { System.out.println("Human sleep.."); } public static void main(String[] args) { Human h = new Male();// 向上轉(zhuǎn)型 System.out.println(h.getName()); } } class Male extends Human { String name = "Male"; public String getName(){ return this.name; } @Override public void sleep() { System.out.println("Male sleep.."); } public void speak() { System.out.println("I am Male"); } } class Female extends Human { String name = "Female"; public String getName(){ return this.name; } @Override public void sleep() { System.out.println("Female sleep.."); } public void speak() { System.out.println("I am Female"); } }
非要調(diào)用子類擴(kuò)展的方法,比如speak方法,就只能向下轉(zhuǎn)型了。
例子:向下轉(zhuǎn)型
向下轉(zhuǎn)型需要考慮安全性,如果父類引用的對象是父類本身,那么在向下轉(zhuǎn)型的過程中是不安全的,編譯不會出錯(cuò),但是運(yùn)行時(shí)會出現(xiàn)java.lang.ClassCastException錯(cuò)誤。它可以使用instanceof來避免出錯(cuò)此類錯(cuò)誤即能否向下轉(zhuǎn)型,只有先經(jīng)過向上轉(zhuǎn)型的對象才能繼續(xù)向下轉(zhuǎn)型。
package multistate; public class Human { public void sleep() { System.out.println("Human sleep.."); } public static void main(String[] args) { Human h = new Male();// 向上轉(zhuǎn)型 Human h1 = new Human(); //h.speak();此時(shí)需要向下轉(zhuǎn)型,否則不能調(diào)用speak方法。 Male m = (Male) h; m.speak(); /**Male m1 = (Male)h1; m1.speak(); 此時(shí)會出現(xiàn)運(yùn)行時(shí)錯(cuò)誤,所以可以用instanceOF判斷*/ if (h1 instanceof Male){ Male m1 = (Male)h1; m1.speak(); } } } class Male extends Human { @Override public void sleep() { System.out.println("Male sleep.."); } public void speak() { System.out.println("I am Male"); } }
弄了半天,向上轉(zhuǎn)型反而不能擁有子類的全部方法,還得向下轉(zhuǎn)型,那直接Son s = new Son();豈不是很方便?不知道是不是就我一個(gè)開始學(xué)習(xí)轉(zhuǎn)型有這種想法。
最后搞明白了,原因還是我我的例子太簡單,沒有 考慮過要把類的對象傳遞給其他函數(shù)的例子。
例子:體現(xiàn)向上轉(zhuǎn)型的好處,節(jié)省代碼。
package multistate; public class Human { public void sleep() { System.out.println("Human sleep.."); } public static void doSleep(Human h){ h.sleep(); }//此時(shí)傳遞的參數(shù)是父類對象,但是實(shí)際調(diào)用時(shí)傳遞子類對象,就是向上轉(zhuǎn)型。 public static void main(String[] args) { Human h = new Male();// 向上轉(zhuǎn)型 doSleep(new Male());//此處匿名子類對象,當(dāng)然實(shí)際應(yīng)用時(shí)應(yīng)該是用上面的向上轉(zhuǎn)型公式,然后將子類對象傳遞進(jìn)來,這樣以后好在向下轉(zhuǎn)型,此處沒有向下轉(zhuǎn)型,所以直接用了匿名類對象。 doSleep(new Female()); } } 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.."); } }
如果不向上轉(zhuǎn)型則必須寫兩個(gè)doSleep函數(shù),一個(gè)傳遞Male類對象,一個(gè)傳遞Female類對象。這還是兩個(gè)子類,如果有很多子類呢,就要寫很多相同的函數(shù),造成重復(fù)。
好,終于也理解了為什么要向上轉(zhuǎn)型,一旦向上轉(zhuǎn)型了,當(dāng)需要用到子類的方法時(shí),就需要向下轉(zhuǎn)型,也就是為什么要向下轉(zhuǎn)型也解決了。
總結(jié)一下:
1、把子類對象直接賦給父類引用叫upcasting向上轉(zhuǎn)型,向上轉(zhuǎn)型不用強(qiáng)制轉(zhuǎn)型。
如Father father = new Son();
2、把指向子類對象的父類引用賦給子類引用叫向下轉(zhuǎn)型(downcasting),要強(qiáng)制轉(zhuǎn)型,要向下轉(zhuǎn)型,必須先向上轉(zhuǎn)型為了安全可以用instanceof判斷。
如father就是一個(gè)指向子類對象的父類引用,把father賦給子類引用son 即Son son =(Son)father;
其中father前面的(Son)必須添加,進(jìn)行強(qiáng)制轉(zhuǎn)換。
3、upcasting 會丟失子類特有的方法,但是子類overriding 父類的方法,子類方法有效,向上轉(zhuǎn)型只能引用父類對象的屬性,要引用子類對象屬性,則要寫getter函數(shù)。
4、向上轉(zhuǎn)型的作用,減少重復(fù)代碼,父類為參數(shù),調(diào)有時(shí)用子類作為參數(shù),就是利用了向上轉(zhuǎn)型。這樣使代碼變得簡潔。體現(xiàn)了JAVA的抽象編程思想。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
java正則表達(dá)式校驗(yàn)日期格式實(shí)例代碼
如果使用得當(dāng),正則表達(dá)式是匹配各種模式的強(qiáng)大工具,下面這篇文章主要給大家介紹了關(guān)于java正則表達(dá)式校驗(yàn)日期格式的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-05-05Servlet實(shí)現(xiàn)共享數(shù)據(jù)JavaWeb組件的幾種方法
本文將結(jié)合實(shí)例代碼,介紹Servlet實(shí)現(xiàn)共享數(shù)據(jù)JavaWeb組件的幾種方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-07-07Java實(shí)現(xiàn)快速將HTML表格轉(zhuǎn)換成Excel
這篇文章主要為大家詳細(xì)介紹一種使用Java的快速將Web中表格轉(zhuǎn)換成Excel的方法,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-05-05Failed to execute goal org...的解決辦法
這篇文章主要介紹了Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.1的解決辦法的相關(guān)資料,需要的朋友可以參考下2017-06-06詳解SpringBoot健康檢查的實(shí)現(xiàn)原理
這篇文章主要介紹了詳解SpringBoot健康檢查的實(shí)現(xiàn)原理,幫助大家更好的理解和學(xué)習(xí)使用SpringBoot框架,感興趣的朋友可以了解下2021-03-03Java實(shí)現(xiàn)年獸大作戰(zhàn)游戲詳解
春節(jié)要到了,看慣了前端各種小游戲,確實(shí)做得很好,很精致。本文將為大家介紹一款java版本的年獸大作戰(zhàn)游戲,感興趣的小伙伴可以試一試2022-01-01深入學(xué)習(xí)Java單元測試(Junit+Mock+代碼覆蓋率)
在做單元測試時(shí),代碼覆蓋率常常被拿來作為衡量測試好壞的指標(biāo),甚至,用代碼覆蓋率來考核測試任務(wù)完成情況,比如,代碼覆蓋率必須達(dá)到80%或 90%。下面我們就來詳細(xì)學(xué)習(xí)下java單元測試吧2019-06-06