Java 多態

来源:https://www.cnblogs.com/guohaohui/archive/2023/02/01/17082362.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("每周讀一本好書");
    }
}

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

-Advertisement-
Play Games
更多相關文章
  • 簡單又高大上的項目 圖形識別、自然語言處理(語言識別、語音轉文字)、文字識別、區塊鏈 1.java實現一個基本的文字識別 引入依賴 <!-- ai 文字識別 --> <dependency> <groupId>com.baidu.aip</groupId> <artifactId>java-sdk< ...
  • 本文介紹基於Python語言,實現對多個不同Excel文件進行數據讀取與平均值計算的方法。 首先,讓我們來看一下具體需求:目前有一個文件夾,其中存放了大量Excel文件;文件名稱是每一位同學的名字,即文件名稱沒有任何規律。 而每一個文件都是一位同學對全班除了自己之外的其他同學的各項打分,我們以其中一 ...
  • 本文描述的是查找字典的某一個元素(字典遍歷元素請點擊->這裡) 上下文代碼 smart_girl = {"name":"yuan wai", "age": 25,"sex":"女"} 第一種方式:[] 註意:這種方式,如果找不到對應的key,會報一個KeyError錯誤 smart_girl["na ...
  • 日期類 一、第一代日期類 Date Date:第一代日期類,精確到毫秒,代表特定的瞬間。 SimpleDateFormat:格式化和解析日期的具體類。它允許進行格式化(日期 -> 文本)、解析(文本 -> 日期)和規範化。 SimpleDateFormat日期-時間格式模式參數: | Letter ...
  • 1. 前言 WebMvcConfigurer配置類其實是Spring內部的一種配置方式,採用JavaBean的形式來代替傳統的xml配置文件形式進行針對框架個性化定製,可以自定義一些Handler,Interceptor,ViewResolver,MessageConverter。基於java-ba ...
  • 使用apidoc包生成apidoc的json格式數據,然後使用python讀取出介面地址、名字、組名、輸入參數格式和例子、輸出參數格式和例子等,然後根據swagger格式填入對應的數據即可生成swagger的json格式 ...
  • 前言 用於實現通過牌子逆向查主播信息這個功能。 插件基於Nonebot2開發,鏈接:https://github.com/Ikaros-521/nonebot_plugin_searchBiliInfo 工程下載 github:https://github.com/Ikaros-521/get_bi ...
  • 首先要瞭解的是,1、功能變數名稱註冊 2、功能變數名稱解析,是兩個獨立的產品。一般情況下,功能變數名稱服務商(萬網、新網等)會提供一站式服務,既提供“功能變數名稱購買註冊”,又提供“功能變數名稱解析服務”。 但實際上,功能變數名稱和功能變數名稱解析是可以分開部署的,功能變數名稱服務商也支持相關的分離設置。比如:功能變數名稱在萬網進行管理,功能變數名稱解析可以指向其他功能變數名稱服務商... ...
一周排行
    -Advertisement-
    Play Games
  • 就像 Web Api 介面可以對入參進行驗證,避免用戶傳入非法的或者不符合我們預期的參數一樣,選項也可以對配置源的內容進行驗證,避免配置中的值與選項類中的屬性不對應或者不滿足預期,畢竟大部分配置都是通過字元串的方式,驗證是很有必要的。 1. 註解驗證 像入參驗證一樣,選項驗證也可以通過特性註解方便地 ...
  • 原文作者:aircraft 原文鏈接:https://www.cnblogs.com/DOMLX/p/17270107.html 加工的泛型類如下: using System; using System.Collections.Generic; using System.IO; using Syst ...
  • 在前一篇文章,我們瞭解瞭如何通過.NET6+Quartz開發基於控制台應用程式的定時任務,今天繼續在之前的基礎上,進一步講解基於ASP.NET Core MVC+Quartz實現定時任務的可視化管理頁面,僅供學習分享使用,如有不足之處,還請指正。 涉及知識點 Quartz組件,關於Quartz組件的 ...
  • 面向對象1 面向對象,更在乎的結果,而過程的實現並不重要 IDea快捷鍵(基礎版) | 快捷鍵 | 作用 | | | | | ctrl + / | 快捷註釋 | | ctrl + shift + / | 多行註釋 | | ctrl + d | 快速複製 | | ctrl + shift + up/d ...
  • NX中的checkmate功能是用於檢查模型、圖紙數據的工具,在UGOPEN中有例子。手動操作可以檢查已載入的裝配下所有零部件,可以設置通過後保存模型,檢查結果保存到Teamcenter中,預設保存在零組件版本下。 代碼中可以設置多個檢查規則。相關設置可以在用戶預設設置中進行設置。 1 // 2 / ...
  • JavaSE 運算符 算術運算符:+,-,*,/,%,++(自增),--(自減) i++:先用後+1;++i:先+1後用 賦值運算符:= 擴展賦值運算符:+=,-=,*=,/= a+=b >a=a+b: ​ 可讀性差,但是編譯效率高,且會自動進行類型轉換; ​ 當ab為基本數據類型時,a+b和b+a ...
  • 面向對象2 訪問修飾符 | | private | default | protected | public | | | | | | | | 當前類 | :heavy_check_mark: | :heavy_check_mark: | :heavy_check_mark: | :heavy_che ...
  • 推薦一些學習qml教程 Qt官方的QML教程: https://doc.qt.io/qt-5/qtqml-index.html 這是一個由Qt官方提供的完整的QML教程,包含了所有基本知識和高級語法。 QML中文網:http://www.qmlcn.com/ 這是一個非常不錯的中文QML學習網站,提 ...
  • QAbstractBUtton: 所有按鈕控制項的基類 提供按鈕的通用功能 繼承自QWidget 屬於抽象類別,不能直接去使用,必須藉助於子類(除非你覺得子類不夠用,想自定義一個按鈕) 大部分功能之前已經使用過,在這裡只作簡單介紹 文本設置: setText(str) :設置按鈕提示文本 text() ...
  • 使用 VLD 記憶體泄漏檢測工具輔助開發時整理的學習筆記。本篇介紹 VLD 配置文件中配置項 StartDisabled 的使用方法。 ...