Java 重载方法的匹配规则-含有变长参数方法的匹配
本文用一个例子来展示 含有变长参数的重载方法是如何匹配的
public class demo {
void invoke(Object obj, Object... args) {
System.out.println(1);
}
void invoke(String s, Object obj, Object... args) {
System.out.println(2);
}
public static void main(String[] args) {
demo d = new demo();
d.invoke(null); //1
d.invoke(null, 1); //2
d.invoke(null,2,1); //2
d.invoke(1,2,1); //1
}
}
重载方法的匹配规则:
——引用自《深入拆解 Java 虚拟机》
重载的方法在编译过程中即可完成识别。具体到每一个方法调用,Java 编译器会根据所传入参数的声明类型(注意与实际类型区分)来选取重载方法。选取的过程其实分为三个阶段:
-
在不考虑对基本类型自动装拆箱(auto-boxing,auto-unboxing),以及可变长参数的情况下选取重载方法;
-
如果在第 1 个阶段中没有找到适配的方法,那么在允许自动装拆箱,但不允许可变长参数的情况下选取重载方法;
-
如果在第 2 个阶段中没有找到适配的方法,那么在允许自动装拆箱以及可变长参数的情况下选取重载方法。
PS:如果 Java 编译器在同一个阶段中找到了多个适配的方法,那么它会在其中选择一个最为贴切 的,而决定贴切程度的一个关键就是形式参数类型的继承关系。
举个例子:当传入 null 时,它既可以匹配第一个方法中声明为 Object 的形式参数,也可 以匹配第二个方法中声明为 String 的形式参数。由于 String 是 Object 的子类,因此 Java 编译 器会认为第二个方法更为贴切。
结合例子
根据以上理论,结合例子可以看到:
- d.invoke(null);方法执行时满足1阶段的匹配规则,所以打印结果为 1;
- d.invoke(null,1);方法执行时满足1阶段的匹配规则,故打印结果为 2;
- d.invoke(null,2,1);方法执行时满足阶段2阶段的匹配规则,所以打印 2
- d.invoke(1,2,1);方法执行时满足阶段2阶段的匹配规则,同时第一个参数无法匹配 String类型,故打印 1
无法精确匹配到方法时
Java 编译器可能无法决定应该调用哪 个目标方法。
在这种情况下,编译器会报错,并且提示这个方法调用有二义性如图所示: