面試突擊:方法優先調用可選參數還是固定參數?
作者:磊哥
來源 | Java面試真題解析(ID:aimianshi666)
轉載請聯系授權(微信ID:GG_Stone)
面試合集:https://gitee.com/mydb/interview
可選參數(varargs)是 JDK 5 中新增的特性,也叫變長參數或可變參數。它是指一個方法的參數中可以用“...”來表示此方法可以接受無窮個參數,這種表示方法就叫可選參數。可選參數的語法如下:
- public void method(數據類型... 參數名稱){
- // 方法體
- }
可選參數基本用法如下:
- public class ArgumentExample {
- public static void main(String[] args) {
- // 調用可選參數
- method("Java");
- System.out.println();
- // 調用可選參數
- method("MySQL", "Redis");
- // 調用可選參數
- System.out.println();
- method("Spring", "Spring MVC", "Spring Boot");
- }
- /**
- * 可選參數方法
- */
- public static void method(String... names) {
- for (String item : names) {
- System.out.println(item);
- }
- }
- }
以上程序的執行結果如下圖所示:
固定參數
固定參數的概念恰好與可選參數相反,固定參數也就是普通的參數,一個方法中有固定的參數類型和個數且沒有“...”修飾就是固定參數。JDK 5 之前所有的方法傳參都是固定參數,如下代碼所示:
- public class ArgumentExample {
- public static void main(String[] args) {
- method("Java");
- }
- /**
- * 固定參數方法
- */
- public static void method(String name) {
- System.out.println("固定參數:" + name);
- }
- }
可選參數注意事項
可選參數在使用時要注意以下 4 個問題。
1.可選參數是從0到無窮
可選參數的調用個數是從 0 到無窮,而不是從 1 到無窮,這點需要注意一下,如以下代碼所示:
- public class ArgumentExample {
- public static void main(String[] args) {
- method();
- }
- /**
- * 可選參數方法
- */
- public static void method(String... names) {
- System.out.println("可選參數數量:" + names.length);
- }
- }
以上程序的執行結果如下圖所示:
從上述代碼可以看出,可選參數即使不傳遞任何參數,也就是 0 個參數,也是能正常調用到的。
2.一個方法只能有一個可選參數
一個方法中只能有一個可選參數,如果有多個可選參數程序會報錯,如下圖所示:
3.可選參數必須要放在方法最后
可選參數如果不放在方法參數的最后面,那么編譯器也會報錯,如下圖所示:
4.可選參數和其他同名方法組成方法重載
可選參數和其他的同名方法可以并存,并且它們組成了方法重載,如下代碼所示:
優先調用固定參數還是可選參數?
基本知識點介紹完,接下來咱們回到本文的主題,當一個方法中有兩類參數:固定參數和可選參數時,究竟是先調用固定參數呢?還是先調用可選參數呢?接下來咱們使用一段代碼來測試一下:
- public class ArgumentExample {
- public static void main(String[] args) {
- method("磊哥聊編程");
- }
- /**
- * 固定參數方法
- */
- public static void method(String name) {
- System.out.println("調用固定參數:" + name);
- }
- /**
- * 可選參數方法
- */
- public static void method(String... names) {
- System.out.println("調用可選參數:" + names.length);
- }
- }
以上程序的執行結果如下:
結論
從上面的結果可以看出,當程序中有固定參數和可選參數時,優先調用固定參數,而非可選參數。
原因分析
看到這,可能有朋友已經意識到了,如果你看過我上一篇《為什么不同返回類型不算方法重載?》就全明白了,究竟是先調用可選參數還是固定參數?上一篇文章在介紹方法重載調用的優先級規則里已經說過了:其中可選參數的調用優先級是最低的,在固定參數和可選參數之間還有其他的調用選項。因為有些朋友沒注意到,或者沒看到,所以我這里再簡單的回顧一下。
第1優先級:精準參數匹配
方法重載會優先調用和方法參數類型一模一樣的固定參數方法。
第2優先級:調用基本類型自動轉換成更大的基本類型
如果是基本數據類型,那么方法重載調用的第 2 匹配原則是自動轉換成更大的基本數據類型,如以下代碼:
- 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) { // 可選參數
- System.out.println("調用 int... 方法");
- }
- }
以上程序的執行結果如下圖所示:
第3優先級:自動裝/拆箱匹配
如果存在基本類型對應的包裝類型,或者是包裝類型對應的基本類型的方法重載,那么會優先調用自動裝箱或自動拆箱的方法重載,如以下代碼所示:
- 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) { // 可選參數
- System.out.println("調用 int... 方法");
- }
- }
以上程序的執行結果如下圖所示:
第4優先級:按照繼承路線依次向上匹配父類
當有父類參數時會優先調用父類重載方法,如下代碼所示:
- 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) { // 可選參數
- System.out.println("調用 int... 方法");
- }
- }
以上程序的執行結果如下圖所示:
第5優先級:可選參數匹配
可選參數的調用優先級是最低的,當一個類中只有可選參數方法時,才會調用可選參數方法。
總結
可選參數是 JDK 5 中新增的以“...”格式存在的參數類型,可選參數可以匹配 0 到無窮個參數,但一個方法中只能有一個可選參數,且可選參數要放在方法參數的最后面。它可以和固定參數組成方法重載,但可選參數的調用優先級是最低的。