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