Java 重载方法的匹配规则-含有变长参数方法的匹配

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 编译器会根据所传入参数的声明类型(注意与实际类型区分)来选取重载方法。选取的过程其实分为三个阶段:

  1. 在不考虑对基本类型自动装拆箱(auto-boxing,auto-unboxing),以及可变长参数的情况下选取重载方法;

  2. 如果在第 1 个阶段中没有找到适配的方法,那么在允许自动装拆箱,但不允许可变长参数的情况下选取重载方法;

  3. 如果在第 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 编译器可能无法决定应该调用哪 个目标方法。

在这种情况下,编译器会报错,并且提示这个方法调用有二义性如图所示:
image

posted @ 2020-04-26 17:04  上帝爱吃苹果-Soochow  阅读(426)  评论(0编辑  收藏