Java全面分析面向?qū)ο笾鄳B(tài)
多態(tài)的理解
什么是多態(tài)呢??從字面理解就是多種形態(tài),也就是不同類實例化出來的對象調(diào)用同一種方法,也可以理解為不同類的對象經(jīng)過同一種行為產(chǎn)生的狀態(tài)是不同的,這就是多態(tài)。
要想理解多態(tài),我們必須要了解向上轉(zhuǎn)型和重寫這兩個重點然后在來深刻理解多態(tài)這一概念,等看完向上轉(zhuǎn)型與重寫再來看多態(tài)的概念,你就會豁然開朗,一下就通透了不少。因為多態(tài)的條件就是向上轉(zhuǎn)型,重寫以及繼承。
向上轉(zhuǎn)型
首先多態(tài)的前提是繼承,那既然是繼承,那就肯定就有父類與子類這樣的關(guān)系。
我們再來回憶一下怎么創(chuàng)建子類對象和父類對象。
class Animal{ public String name;//名字 public int age; public void eat() { System.out.println("我要吃飯!?。?); } public void sleep() { System.out.println("我要睡覺?。?!"); } } class Cat extends Animal{ public void mew() { System.out.println("喵喵喵?。?!"); } } public class TestDemo1 { public static void main(String[] args) { Cat cat =new Cat();//實例化子類對象 cat.name="mimi"; Animal animal = new Animal();//實例化父類對象 animal.eat(); } }
這里就創(chuàng)建了貓這個類然后繼承了Animal類。我們實例化貓和Animal這個對象就可以調(diào)用方法和屬性。
那何為向上轉(zhuǎn)型呢???
原本子類對象的引用引用子類的對象,現(xiàn)在讓父類的引用引用子類對象這就是向上轉(zhuǎn)型。
我們利用代碼理解一下:
這就是向上轉(zhuǎn)型,我們也可以利用animal這個父類引用 調(diào)用方法;
這時我們就會發(fā)現(xiàn)利用這個引用能夠調(diào)用父類的方法和屬性,但是不能夠調(diào)用子類的方法和屬性,那為什么呢??原因就是因為父類沒有子類這個方法,所以不能調(diào)用??偨Y(jié):向上轉(zhuǎn)型的時候也就是父類引用引用子類對象,這個父類引用只能調(diào)用父類有的屬性和方法,不能調(diào)用子類的。
向上轉(zhuǎn)型的三種形式
第一種:直接賦值
也就是我們上面的那種寫法:
Animal animal1 = new Cat();//父類對象的引用 引用子類對象--->向上轉(zhuǎn)型 Animal animal2 = new Dog();
第二種:作為方法參數(shù):
第三種作為返回值:
我們回到剛才的打印結(jié)果是什么;
但如果我把父類的方法變成我要吃貓糧呢?那結(jié)果毫無意外就是mimi我要吃貓糧。
但是這就會出現(xiàn)一個問題,如果我在創(chuàng)建一個狗類,然后在調(diào)用eat方法 難道狗也要吃貓糧么?這就會出現(xiàn)問題,那我們可以在子類寫一個eat方法;
class Animal{ public String name;//名字 public int age; public void eat() { System.out.println(this.name+"要吃飯?。?!"); } } class Dog extends Animal{ public void dark() { System.out.println("汪汪汪?。?!"); } public void eat() { System.out.println(this.name+"吃狗糧?。?!"); } } class Cat extends Animal{ public void mew() { System.out.println("喵喵喵!??!"); } public void eat() { System.out.println(this.name+"吃貓糧!!!"); } } public class TestDemo1 { public static void main(String[] args) { //語法形式 : 父類 變量 = new 子類(); Animal animal1 = new Cat();//父類對象的引用 引用子類對象--->向上轉(zhuǎn)型 Animal animal2 = new Dog();//父類對象的引用 引用子類對象--->向上轉(zhuǎn)型 animal1.name = "小貓";//訪問父類屬性 animal2.name = "小狗";//訪問父類屬性 animal1.eat(); animal2.eat(); // animal.mew();//訪問子類特有的方法 } }
這時又創(chuàng)建了一個狗類,然后又分別在兩個子類創(chuàng)建兩個eat方法。
我們發(fā)現(xiàn)這時候就變得很清楚就達(dá)到我們想要的效果了。
但我們又應(yīng)該想一想,為什么調(diào)用子類的eat方法而不調(diào)用父類的?
動態(tài)綁定和靜態(tài)綁定
此時其實發(fā)生了動態(tài)綁定,我們可以看一下字節(jié)碼文件,打開powershell窗口
我們都知道執(zhí)行一個程序是先編譯后運行,而這個是在編譯的時候調(diào)用的是Animal的eat方法,而在運行的時候是調(diào)用的是Cat的方法這就是我們所說的運行時綁定或者可以說是動態(tài)綁定。
那既然有動態(tài)綁定那肯定也有靜態(tài)綁定。
動態(tài)綁定是在編譯的時候調(diào)用一個方法,而在運行時才是最后要確定調(diào)用的方法,也就是在運行時確定要調(diào)用那個方法。
靜態(tài)綁定就是在編譯期間已經(jīng)確定要調(diào)用哪個方法。
其中,動態(tài)綁定最顯著的代表就是方法重寫。
靜態(tài)綁定最顯著的代表就是方法重載。
我們在回過頭看上面的方法 ε=(´ο`*)))......怎么前面的eat方法返回值,參數(shù)列表,方法名都是一樣的呢?我們來看一下。
方法的重寫
我們之前學(xué)過方法重載這里回顧一下方法重載,方法重載是方法名相同,返回值不做要求,參數(shù)列表不同。而我們今天學(xué)的方法重寫是返回值相同,方法名稱相同,參數(shù)列表相同,說是叫方法重寫其實也可以叫做方法覆蓋。
方法重寫有幾點注意要求:
方法重寫滿足 方法名相同,方法的參數(shù)列表相同,方法的返回值相同。
我們也可以一鍵生成重寫
有幾個注意事項:
不能重寫被private修飾的方法。
不能重寫被final修飾的方法。
子類的方法的訪問權(quán)限一定要大于等于父類的訪問權(quán)限。
重寫的方法, 可以使用 @Override 注解來顯式指定. 有了這個注解能幫我們進(jìn)行一些合法性校驗. 例如不小心將方法名字拼寫錯了 (比如寫成eat), 那么此時編譯器就會發(fā)現(xiàn)父類中沒有 aet 方法, 就會編譯報錯, 提示無法構(gòu)成重寫.
被static修飾的方法也不能被重寫
總結(jié)方法重寫的注意事項:
- 被private,final修飾的方法不能被重寫。
- 被staitc修飾的方法也不能被重寫。
- @override 可以檢查你重寫的方法名是否正確,最好要加上。
- 方法重寫一定滿足方法名相同,參數(shù)列表相同,返回值相同。
對比方法重寫與方法重載:
最后:重寫不是進(jìn)行在原來基礎(chǔ)的修改,而是在原來基礎(chǔ)上進(jìn)行迭代和更新。
進(jìn)一步認(rèn)識和理解多態(tài)
場景:畫一個圖形
class Shape{//創(chuàng)建一個圖形類---->作為多種圖形的父類 public int length;//圖形的長 public int wide;//圖形的寬 public int height;//圖形的高 public void draw() { System.out.println("我要畫一個圖形?。?!"); } } class rectangle extends Shape{//長方形 @Override public void draw() { System.out.println("我要畫一個長方形?。?!"); } } class square extends Shape{ @Override public void draw() { System.out.println("我要畫一個正方形!??!"); } } class circular extends Shape{ @Override public void draw() { System.out.println("我要畫一個圓形?。?!"); } } public class TestDemo1 { public static void method(Shape shape) { shape.draw(); } public static void main(String[] args) { Shape shape1 = new circular(); Shape shape2 = new rectangle(); Shape shape3 = new square(); method(shape1); method(shape2); method(shape3); } }
創(chuàng)建一個Shape(父類),然后創(chuàng)建三個子類分別是square ,circular,rectangle,利用父類引用這三個子類,接著調(diào)用method方法。
這就是多態(tài),不同的對象,調(diào)用同一個方法最后結(jié)果產(chǎn)生出不同的狀態(tài)。
我們再來總結(jié)多態(tài)產(chǎn)生的條件:
- 要在繼承體系下
- 子類要對父類的方法進(jìn)行重寫
- 通過父類的引用調(diào)用重寫的方法
也就是 在繼承體系下 進(jìn)行向上轉(zhuǎn)型 和 方法重寫
多態(tài)的優(yōu)點
優(yōu)點:
- 能夠降低代碼的 "圈復(fù)雜度", 避免使用大量的 if - else
- 如果使用多態(tài), 則不必寫這么多的 if - else 分支語句, 代碼更簡單.
- 可擴(kuò)展能力更強
缺點:
- 代碼的運行效率降低
還有一個重要點就是不要在構(gòu)造方法中調(diào)用重寫方法
到此這篇關(guān)于Java全面分析面向?qū)ο笾鄳B(tài)的文章就介紹到這了,更多相關(guān)Java多態(tài)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java畢業(yè)設(shè)計實戰(zhàn)之養(yǎng)老院管理系統(tǒng)的實現(xiàn)
讀萬卷書不如行萬里路,只學(xué)書上的理論是遠(yuǎn)遠(yuǎn)不夠的,只有在實戰(zhàn)中才能獲得能力的提升,本篇文章手把手帶你用java+SSM+JSP+Easyui+maven+mysql實現(xiàn)一個養(yǎng)老院管理系統(tǒng),大家可以在過程中查缺補漏,提升水平2022-03-03springboot oauth2實現(xiàn)單點登錄實例
我們見過的很多網(wǎng)站,容許使用第三方賬號登錄,oauth2是用來做三方登錄的,本文就詳細(xì)的介紹springboot oauth2實現(xiàn)單點登錄實例,具有一定的參考價值,感興趣的可以了解一下2022-01-01Java excel數(shù)據(jù)導(dǎo)入mysql的實現(xiàn)示例詳解
今天教大家如何使用Java將excel數(shù)據(jù)導(dǎo)入MySQL,文中有非常詳細(xì)的代碼示例,對正在學(xué)習(xí)java的小伙伴呢很有幫助,需要的朋友可以參考下2022-08-08詳解IDEA使用Maven項目不能加入本地Jar包的解決方法
這篇文章主要介紹了詳解IDEA使用Maven項目不能加入本地Jar包的解決方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08使用JAXBContext 設(shè)置xml節(jié)點屬性
這篇文章主要介紹了使用JAXBContext 設(shè)置xml節(jié)點屬性的操作,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-08-08