Java:多態乃幸福本源

来源:https://www.cnblogs.com/qing-gee/archive/2018/12/14/10117431.html
-Advertisement-
Play Games

在我刻板的印象里,西游記里的那段孫悟空和二郎神的精彩對戰就能很好的解釋“多態”這個詞:一個孫悟空,能七十二變;一個二郎神,也能七十二變;他們都可以變成不同的形態,但只需要悄悄地喊一聲“變”。 ...


01 多態是什麼

在我刻板的印象里,西游記里的那段孫悟空和二郎神的精彩對戰就能很好的解釋“多態”這個詞:一個孫悟空,能七十二變;一個二郎神,也能七十二變;他們都可以變成不同的形態,但只需要悄悄地喊一聲“變”。

Java的多態是什麼呢?其實就是一種能力——同一個行為具有不同的表現形式;換句話說就是,執行一段代碼,Java在運行時能根據對象的不同產生不同的結果。和孫悟空和二郎神都只需要喊一聲“變”,然後就變了,並且每次變得還不一樣;一個道理。

多態的前提條件有三個:

  • 子類繼承父類
  • 子類覆蓋父類的方法
  • 父類引用指向子類對象

多態的一個簡單應用,來看程式清單1-1:

//子類繼承父類
public class Wangxiaoer extends Wanger {
	public void write() { // 子類覆蓋父類方法
		System.out.println("記住仇恨,表明我們要奮發圖強的心智");
	}

	public static void main(String[] args) {
		// 父類引用指向子類對象
		Wanger[] wangers = { new Wanger(), new Wangxiaoer() };

		for (Wanger wanger : wangers) {
			// 對象是王二的時候輸出:勿忘國恥
			// 對象是王小二的時候輸出:記住仇恨,表明我們要奮發圖強的心智
			wanger.write();
		}
	}
}

class Wanger {
	public void write() {
		System.out.println("勿忘國恥");
	}
}

02 多態與後期綁定

現在,我們來思考一個問題:程式清單1-1在執行wanger.write()時,由於編譯器只有一個Wanger引用,它怎麼知道究竟該調用父類Wanger的write()方法,還是子類Wangxiaoer的write()方法呢?

答案是在運行時根據對象的類型進行後期綁定,編譯器在編譯階段並不知道對象的類型,但是Java的方法調用機制能找到正確的方法體,然後執行出正確的結果。

多態機制提供的一個重要的好處程式具有良好的擴展性。來看程式清單2-1:

//子類繼承父類
public class Wangxiaoer extends Wanger {
	public void write() { // 子類覆蓋父類方法
		System.out.println("記住仇恨,表明我們要奮發圖強的心智");
	}
	
	public void eat() {
		System.out.println("我不喜歡讀書,我就喜歡吃");
	}

	public static void main(String[] args) {
		// 父類引用指向子類對象
		Wanger[] wangers = { new Wanger(), new Wangxiaoer() };

		for (Wanger wanger : wangers) {
			// 對象是王二的時候輸出:勿忘國恥
			// 對象是王小二的時候輸出:記住仇恨,表明我們要奮發圖強的心智
			wanger.write();
		}
	}
}

class Wanger {
	public void write() {
		System.out.println("勿忘國恥");
	}
	
	public void read() {
		System.out.println("每周讀一本好書");
	}
}

在程式清單2-1中,我們在Wanger類中增加了read()方法,在Wangxiaoer類中增加了eat()方法,但這絲毫不會影響到write()方法的調用。write()方法忽略了周圍代碼發生的變化,依然正常運行。這讓我想起了金庸《倚天屠龍記》里九陽真經的口訣:“他強由他強,清風拂山崗;他橫由他橫,明月照大江。”

多態的這個優秀的特性,讓我們在修改代碼的時候不必過於緊張,因為多態是一項讓程式員“將改變的與未改變的分離開來”的重要特性。

03 多態與構造器

在構造器中調用多態方法,會產生一個奇妙的結果,我們來看程式清單3-1:

public class Wangxiaosan extends Wangsan {
	private int age = 3;
	public Wangxiaosan(int age) {
		this.age = age;
		System.out.println("王小三的年齡:" + this.age);
	}
	
	public void write() { // 子類覆蓋父類方法
		System.out.println("我小三上幼兒園的年齡是:" + this.age);
	}
	
	public static void main(String[] args) {
		new Wangxiaosan(4);
//		上幼兒園之前
//		我小三上幼兒園的年齡是:0
//		上幼兒園之後
//		王小三的年齡:4
	}
}

class Wangsan {
	Wangsan () {
		System.out.println("上幼兒園之前");
		write();
		System.out.println("上幼兒園之後");
	}
	public void write() {
		System.out.println("老子上幼兒園的年齡是3歲半");
	}
}

從輸出結果上看,是不是有點詫異?明明在創建Wangxiaosan對象的時候,年齡傳遞的是4,但輸出結果既不是“老子上幼兒園的年齡是3歲半”,也不是“我小三上幼兒園的年齡是:4”。

為什麼?

因為在創建子類對象時,會先去調用父類的構造器,而父類構造器中又調用了被子類覆蓋的多態方法,由於父類並不清楚子類對象中的屬性值是什麼,於是把int類型的屬性暫時初始化為0,然後再調用子類的構造器(子類構造器知道王小二的年齡是4)。

04 多態與向下轉型

向下轉型是指將父類引用強轉為子類類型;這是不安全的,因為有的時候,父類引用指向的是父類對象,向下轉型就會拋出ClassCastException,表示類型轉換失敗;但如果父類引用指向的是子類對象,那麼向下轉型就是成功的。

來看程式清單4-1:

public class Wangxiaosi extends Wangsi {
	public void write() {
		System.out.println("記住仇恨,表明我們要奮發圖強的心智");
	}

	public void eat() {
		System.out.println("我不喜歡讀書,我就喜歡吃");
	}

	public static void main(String[] args) {
		Wangsi[] wangsis = { new Wangsi(), new Wangxiaosi() };

		// wangsis[1]能夠向下轉型
		((Wangxiaosi) wangsis[1]).write();
		// wangsis[0]不能向下轉型
		((Wangxiaosi)wangsis[0]).write();
	}
}

class Wangsi {
	public void write() {
		System.out.println("勿忘國恥");
	}

	public void read() {
		System.out.println("每周讀一本好書");
	}
}

05 總結

我喜歡把複雜的事情儘量簡單化,把簡單的事情有趣化——多態是Java的三大特性之一,它本來需要長篇大論的介紹,但我覺得實在沒有必要,把關鍵的知識點提煉出來就足夠了。更重要的是,你要通過實踐去感知多態的優秀之處。

Java 技術驛站的chenssy對多態下了一個非常經典的結論,我們不妨大聲的朗讀幾遍:

多態就是指程式中定義的引用變數所指向的具體類型和通過該引用變數發出的方法調用在編譯時並不確定,而是在程式運行期間才確定;即一個引用變數倒底會指向哪個類的實例對象,該引用變數發出的方法調用到底是哪個類中實現的方法,必須在由程式運行期間才能決定。因為在程式運行時才確定具體的類,這樣,不用修改源程式代碼,就可以讓引用變數綁定到各種不同的類實現上,從而導致該引用調用的具體方法隨之改變,即不修改程式代碼就可以改變程式運行時所綁定的具體代碼,讓程式可以選擇多個運行狀態,這就是多態性。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 迭代器模式,Iterator,java集合框架內置的一種模式,本文介紹了迭代器模式的起源含義,設計意圖,以及結構形態,並且給出了Java版本的迭代器模式的實現,迭代器模式分為內部迭代和外部迭代,Java集合框架使用的這種形式是比較好的一種方式。 ...
  • 一. Python程式中, 文件的處理步驟是什麼? 二. 文本打開時設置的模式有哪些? 分別代表什麼意思? 三. os模塊中提供的常用文件操作? 四. 代碼實現: 大文件拷貝操作 註意: 不能一次性讀取大文件內容, 容易造成記憶體峰值 五. 代碼實現: 假設一個文件夾中有很多不同格式的文件, 要求: ...
  • 126.Struts2中的攔截器有什麼用?列舉框架提供的攔截器名稱? 127.Struts2有哪些優點? 128.ActionContext和ValueStack什麼時候創建?是否是線程安全的? 129.一個請求在Struts2框架中的處理大概分為幾個步驟? 130.介紹一下Struts的Actio ...
  • activemq配置jmx 配置activemq中的jmx可以用於監控activemq信息。 activemq.xml配置 修改broker屬性 添加節點managementContext <managementContext> <managementContext createConnector= ...
  • 更新Composer依賴報錯處理 Fatal error: Declaration of Fxp\Composer\AssetPlugin\Repository\AbstractAssetsRe pository::search() must be compatible with Composer\... ...
  • 在 WEB 項目中返回 JSON 數據是常見的交互形式,在 Spring Boot 中這一切都變得十分簡單。So easy!!! 你所需具備的基礎 "什麼是 Spring Boot?" "Spring Boot 核心配置文件詳解" "Spring Boot 開啟的 2 種方式" "Spring Bo ...
  • 簡介 從今天開始,我們嘗試用2篇博客的內容量,搞定一個網站叫做“美空網”網址為:http://www.moko.cc/, 這個網站我分析了一下,我們要爬取的圖片在 下麵這個網址 http://www.moko.cc/post/1302075.html 然後在去分析一下,我需要找到一個圖片列表頁面是最 ...
  • Python 中可以讀取 word 文件的庫有 python-docx 和 pywin32。 pywin32 這個庫很強大,不僅僅可以讀取 word,但是網上介紹用 pywin32 讀取 .doc 的文章真不多,因為,真心不好用。 以下是 pywin32 讀取 .doc 的代碼示例,但是讀取表格有問 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...