Java 轉(zhuǎn)型(向上或向下轉(zhuǎn)型)詳解及簡單實(shí)例
在Java編程中經(jīng)常碰到類型轉(zhuǎn)換,對(duì)象類型轉(zhuǎn)換主要包括向上轉(zhuǎn)型和向下轉(zhuǎn)型。
向上轉(zhuǎn)型
我們在現(xiàn)實(shí)中常常這樣說:這個(gè)人會(huì)唱歌。在這里,我們并不關(guān)心這個(gè)人是黑人還是白人,是成人還是小孩,也就是說我們更傾向于使用抽象概念“人”。再例如,麻雀是鳥類的一種(鳥類的子類),而鳥類則是動(dòng)物中的一種(動(dòng)物的子類)。我們現(xiàn)實(shí)中也經(jīng)常這樣說:麻雀是鳥。這兩種說法實(shí)際上就是所謂的向上轉(zhuǎn)型,通俗地說就是子類轉(zhuǎn)型成父類。這也符合Java提倡的面向抽象編程思想。來看下面的代碼:
package a.b; public class A { public void a1() { System.out.println("Superclass"); } }
A的子類B:
package a.b; public class B extends A { public void a1() { System.out.println("Childrenclass"); //覆蓋父類方法 } public void b1(){} //B類定義了自己的新方法 }
C類:
package a.b; public class C { public static void main(String[] args) { A a = new B(); //向上轉(zhuǎn)型 a.a1(); } }
如果運(yùn)行C,輸出的是Superclass 還是Childrenclass?不是你原來預(yù)期的Superclass,而是Childrenclass。這是因?yàn)閍實(shí)際上指向的是一個(gè)子類對(duì)象。當(dāng)然,你不用擔(dān)心,Java虛擬機(jī)會(huì)自動(dòng)準(zhǔn)確地識(shí)別出究竟該調(diào)用哪個(gè)具體的方法。不過,由于向上轉(zhuǎn)型,a對(duì)象會(huì)遺失和父類不同的方法,例如b1()。有人可能會(huì)提出疑問:這不是多此一舉嗎?我們完全可以這樣寫:
B a = new B(); a.a1();
確實(shí)如此!但這樣就喪失了面向抽象的編程特色,降低了可擴(kuò)展性。其實(shí),不僅僅如此,向上轉(zhuǎn)型還可以減輕編程工作量。來看下面的顯示器類Monitor:
package a.b; public class Monitor{ public void displayText() {} public void displayGraphics() {} }
液晶顯示器類LCDMonitor是Monitor的子類:
package a.b; public class LCDMonitor extends Monitor { public void displayText() { System.out.println("LCD display text"); } public void displayGraphics() { System.out.println("LCD display graphics"); } }
陰極射線管顯示器類CRTMonitor自然也是Monitor的子類:
package a.b; public class CRTMonitor extends Monitor { public void displayText() { System.out.println("CRT display text"); } public void displayGraphics() { System.out.println("CRT display graphics"); } }
等離子顯示器PlasmaMonitor也是Monitor的子類:
package a.b; public class PlasmaMonitor extends Monitor { public void displayText() { System.out.println("Plasma display text"); } public void displayGraphics() { System.out.println("Plasma display graphics"); } }
現(xiàn)在有一個(gè)MyMonitor類。假設(shè)沒有向上轉(zhuǎn)型,MyMonitor類代碼如下:
package a.b; public class MyMonitor { public static void main(String[] args) { run(new LCDMonitor()); run(new CRTMonitor()); run(new PlasmaMonitor()); } public static void run(LCDMonitor monitor) { monitor.displayText(); monitor.displayGraphics(); } public static void run(CRTMonitor monitor) { monitor.displayText(); monitor.displayGraphics(); } public static void run(PlasmaMonitor monitor) { monitor.displayText(); monitor.displayGraphics(); } }
可能你已經(jīng)意識(shí)到上述代碼有很多重復(fù)代碼,而且也不易維護(hù)。有了向上轉(zhuǎn)型,代碼可以更為簡潔:
package a.b; public class MyMonitor { public static void main(String[] args) { run(new LCDMonitor()); //向上轉(zhuǎn)型 run(new CRTMonitor()); //向上轉(zhuǎn)型 run(new PlasmaMonitor()); //向上轉(zhuǎn)型 } public static void run(Monitor monitor) { //父類實(shí)例作為參數(shù) monitor.displayText(); monitor.displayGraphics(); } }
我們也可以采用接口的方式,例如:
package a.b; public interface Monitor { abstract void displayText(); abstract void displayGraphics(); }
將液晶顯示器類LCDMonitor稍作修改:
package a.b; public class LCDMonitor implements Monitor { public void displayText() { System.out.println("LCD display text"); } public void displayGraphics() { System.out.println("LCD display graphics"); } }
CRTMonitor、PlasmaMonitor類的修改方法與LCDMonitor類似,而MyMonitor可以不不作任何修改。
可以看出,向上轉(zhuǎn)型體現(xiàn)了類的多態(tài)性,增強(qiáng)了程序的簡潔性。
向下轉(zhuǎn)型
子類轉(zhuǎn)型成父類是向上轉(zhuǎn)型,反過來說,父類轉(zhuǎn)型成子類就是向下轉(zhuǎn)型。但是,向下轉(zhuǎn)型可能會(huì)帶來一些問題:我們可以說麻雀是鳥,但不能說鳥就是麻雀。來看下面的例子:
A類:
package a.b; public class A { void aMthod() { System.out.println("A method"); } }
A的子類B:
package a.b; public class B extends A { void bMethod1() { System.out.println("B method 1"); } void bMethod2() { System.out.println("B method 2"); } }
C類:
package a.b; public class C { public static void main(String[] args) { A a1 = new B(); // 向上轉(zhuǎn)型 a1.aMthod(); // 調(diào)用父類aMthod(),a1遺失B類方法bMethod1()、bMethod2() B b1 = (B) a1; // 向下轉(zhuǎn)型,編譯無錯(cuò)誤,運(yùn)行時(shí)無錯(cuò)誤 b1.aMthod(); // 調(diào)用父類A方法 b1.bMethod1(); // 調(diào)用B類方法 b1.bMethod2(); // 調(diào)用B類方法 A a2 = new A(); B b2 = (B) a2; // 向下轉(zhuǎn)型,編譯無錯(cuò)誤,運(yùn)行時(shí)將出錯(cuò) b2.aMthod(); b2.bMethod1(); b2.bMethod2(); } }
從上面的代碼我們可以得出這樣一個(gè)結(jié)論:向下轉(zhuǎn)型需要使用強(qiáng)制轉(zhuǎn)換。運(yùn)行C程序,控制臺(tái)將輸出:
Exception in thread "main" java.lang.ClassCastException: a.b.A cannot be cast to a.b.B at a.b.C.main(C.java:14) A method A method B method 1 B method 2
其實(shí)黑體部分的向下轉(zhuǎn)型代碼后的注釋已經(jīng)提示你將發(fā)生運(yùn)行時(shí)錯(cuò)誤。為什么前一句向下轉(zhuǎn)型代碼可以,而后一句代碼卻出錯(cuò)?這是因?yàn)閍1指向一個(gè)子類B的對(duì)象,所以子類B的實(shí)例對(duì)象b1當(dāng)然也可以指向a1。而a2是一個(gè)父類對(duì)象,子類對(duì)象b2不能指向父類對(duì)象a2。那么如何避免在執(zhí)行向下轉(zhuǎn)型時(shí)發(fā)生運(yùn)行時(shí)ClassCastException異常?使用5.7.7節(jié)學(xué)過的instanceof就可以了。我們修改一下C類的代碼:
A a2 = new A(); if (a2 instanceof B) { B b2 = (B) a2; b2.aMthod(); b2.bMethod1(); b2.bMethod2(); }
這樣處理后,就不用擔(dān)心類型轉(zhuǎn)換時(shí)發(fā)生ClassCastException異常了。
感謝閱讀,希望能幫助到大家,謝謝大家對(duì)本站的支持!
相關(guān)文章
SpringBoot整合Kaptcha實(shí)現(xiàn)圖片驗(yàn)證碼加減乘除功能
在開發(fā)Web應(yīng)用時(shí),驗(yàn)證碼是一個(gè)常見的功能,它可以幫助我們防止機(jī)器人的惡意操作,今天我們將學(xué)習(xí)如何使用Kaptcha生成圖片驗(yàn)證碼,并自定義驗(yàn)證碼內(nèi)容為100以內(nèi)的加減乘除運(yùn)算,感興趣的朋友跟隨小編一起看看吧2024-07-07Java并發(fā)編程service層處理并發(fā)事務(wù)加鎖可能會(huì)無效問題
這篇文章主要介紹了Java并發(fā)編程service層處理并發(fā)事務(wù)加鎖可能會(huì)無效問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-07-07springboot實(shí)現(xiàn)SSE(Server?Sent?Event)的示例代碼
SSE?全稱Server?Sent?Event,直譯一下就是服務(wù)器發(fā)送事件,本文主要為大家詳細(xì)介紹了springboot實(shí)現(xiàn)SSE的相關(guān)知識(shí),需要的可以參考一下2024-04-04maven中no main manifest attribute的問題解決
本文主要介紹了maven中no main manifest attribute的問題解決,這個(gè)錯(cuò)誤通常意味著Spring Boot應(yīng)用在啟動(dòng)時(shí)遇到了問題,下面就來具體介紹一下,感興趣的可以了解一下2024-08-08