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
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...