Java中為什么不同的返回類型不算方法重載
方法重載是指在同一個類中,定義了多個同名方法,但每個方法的參數(shù)類型或者是參數(shù)個數(shù)不同就是方法重載。 比如以下 4 個 method 方法就可以稱之為方法重載,
如下代碼所示:
public class OverloadExample { public void method() { // doSomething } public void method(String name) { // doSomething } public void method(Integer id) { // doSomething } public void method(Integer id, String name) { // doSomething } }
為什么不同返回類型不算方法重載?
要回答這個問題,首先要了解一點前置內容,方法簽名。 方法簽名是由:方法名稱 + 參數(shù)類型 + 參數(shù)個數(shù)組成的一個唯一值,這個唯一值就是方法簽名,而 JVM(Java 虛擬機)就是通過這個方法簽名來決定調用哪個方法的。 從方法簽名的組成規(guī)則我們可以看出,方法的返回類型不是方法簽名的組成部分,所以當同一個類中出現(xiàn)了多個方法名和參數(shù)相同,但返回值類型不同的方法時,JVM 就沒辦法通過方法簽名來判斷到底要調用哪個方法了,如下圖所示:
那為什么返回類型不能做為方法簽名的一部分呢? 原因其實很簡單,試想一下,如果方法的返回類型也作為方法簽名的一部分,那么當程序員寫了一個代碼去調用“重載”的方法時,JVM 就不能分辨要調用哪個方法了,
如下代碼所示:
public class OverloadExample { public static void main(String[] args) { OverloadExample example = new OverloadExample(); example.method("磊哥"); // JVM 應該調用哪個方法? } public int method(String name) { // doSomething return 666; } public String method(String name) { // doSomething return "磊哥聊編程"; } }
像以上情況,JVM 就推斷不出來要調用哪個方法了,所以方法的返回類型不能作為方法簽名的一部分。
方法重載的使用場景
方法重載的經典使用場景是 String 類型的 valueOf 方法,valueOf 方法重載有 9 種實現(xiàn),
如下圖所示:
它可以將數(shù)組、對象和基礎數(shù)據類型轉換成字符串類型。
方法重載匹配原則
方法重載的調用順序是有前后之分的,比如以下代碼:
public class OverloadExample { public static void main(String[] args) { OverloadExample example = new OverloadExample(); example.method(12); } public void method(int num) { System.out.println("調用 int 方法"); } public void method(long num) { System.out.println("調用 long 方法"); } public void method(Integer num) { System.out.println("調用 Integer 方法"); } public void method(Object num) { System.out.println("調用 Object 方法"); } public void method(int... num) { // 可選參數(shù) System.out.println("調用 int... 方法"); } }
當出現(xiàn)方法重載時,程序要調用哪個方法呢?執(zhí)行以上程序的執(zhí)行結果如下:
因此我們可以得出以下結論。
匹配原則1:精準類型匹配
方法重載會優(yōu)先調用和方法參數(shù)類型一模一樣的方法,這是第一優(yōu)先匹配原則:精準類型匹配。
匹配原則2:基本類型自動轉換成更大的基本類型
接下來我們把精準匹配方法刪掉,觀察一下第二匹配順序是什么?
實現(xiàn)代碼如下:
public class OverloadExample { public static void main(String[] args) { OverloadExample example = new OverloadExample(); example.method(12); } public void method(long num) { System.out.println("調用 long 方法"); } public void method(Integer num) { System.out.println("調用 Integer 方法"); } public void method(Object num) { System.out.println("調用 Object 方法"); } public void method(int... num) { // 可選參數(shù) System.out.println("調用 int... 方法"); } }
以上程序的執(zhí)行結果如下圖所示:
因此我們可以得出結論:如果是基本數(shù)據類型,那么方法重載調用的第二匹配原則是自動轉換成更大的基本數(shù)據類型。
匹配原則3:自動裝/拆箱匹配
接下來將第二匹配原則中的 long 方法也刪除掉,實現(xiàn)代碼如下:
public class OverloadExample { public static void main(String[] args) { OverloadExample example = new OverloadExample(); example.method(12); } public void method(Integer num) { System.out.println("調用 Integer 方法"); } public void method(Object num) { System.out.println("調用 Object 方法"); } public void method(int... num) { // 可選參數(shù) System.out.println("調用 int... 方法"); } }
以上程序的執(zhí)行結果如下圖所示:
從上述執(zhí)行結果可以看出,方法重載的第三匹配原則是,匹配自動裝箱或拆箱的數(shù)據類型。
匹配原則4:按照繼承路線依次向上匹配
此時將第三匹配原則中的 Integer 方法刪除,剩下代碼如下:
public class OverloadExample { public static void main(String[] args) { OverloadExample example = new OverloadExample(); example.method(12); } public void method(Object num) { System.out.println("調用 Object 方法"); } public void method(int... num) { // 可選參數(shù) System.out.println("調用 int... 方法"); } }
以上程序的執(zhí)行結果如下圖所示:
從上述執(zhí)行結果可以看出,方法重載的第四匹配原則是,依次向上匹配父類的方法調用。
匹配原則5:可變參數(shù)匹配
最后將代碼中的方法刪除的只剩一個可選參數(shù),實現(xiàn)代碼如下:
public class OverloadExample { public static void main(String[] args) { OverloadExample example = new OverloadExample(); example.method(12); } public void method(int... num) { // 可選參數(shù) System.out.println("調用 int... 方法"); } }
以上程序的執(zhí)行結果如下圖所示:
從上述執(zhí)行結果可以看出,方法重載的第五匹配原則是,匹配可選參數(shù)。
總結
在同一個類中定義了多個同名方法,但每個方法的參數(shù)類型或者是參數(shù)個數(shù)不同就是方法重載。方法重載的典型使用場景是 String 中的 valueOf 方法,它有 9 種實現(xiàn)。方法返回類型不能作為方法重載的依據,因為它不是方法簽名的組成部分。方法重載有 5 個匹配原則:精準匹配、基本類型自動轉換成更大的基本類型匹配、自動裝/拆箱匹配、按照繼承路線依次向上匹配、可變參數(shù)匹配。
到此這篇關于Java中為什么不同的返回類型不算方法重載的文章就介紹到這了,更多相關Java返回類型內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
詳解JAVA 線程-線程的狀態(tài)有哪些?它是如何工作的?
這篇文章主要介紹了詳解JAVA 線程的的相關資料,文中講解非常細致,源碼幫助大家更好的理解和學習,感興趣的朋友可以參考下2020-06-06解析Spring Data JPA的Audit功能之審計數(shù)據庫變更
Spring Data JPA 提供了Audit審計功能,用來記錄創(chuàng)建時間、創(chuàng)建人、修改時間、修改人等,下面來詳細講解下審計數(shù)據庫變更2021-06-06SpringBoot中的JPA(Java?Persistence?API)詳解
這篇文章主要介紹了SpringBoot中的JPA(Java?Persistence?API)詳解,JPA用于將?Java?對象映射到關系型數(shù)據庫中,它提供了一種面向對象的方式來操作數(shù)據庫,使得開發(fā)者可以更加方便地進行數(shù)據持久化操作,需要的朋友可以參考下2023-07-07詳解mall整合SpringBoot+MyBatis搭建基本骨架
這篇文章主要介紹了詳解mall整合SpringBoot+MyBatis搭建基本骨架,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2019-08-08Java調用opencv IDEA環(huán)境配置的教程詳解
這篇文章主要為大家詳細介紹了Java調用opencv IDEA環(huán)境配置的相關知識,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學習一下2024-03-03