Java:重载方法解析和varargs - 令人困惑的例子

Jas*_*n S 13 java variadic-functions jls

正当我认为我理解JLS15.12应用于varargs时,这是这个例子:

package com.example.test.reflect;

public class MethodResolutionTest2 {
    public int compute(Object obj1, Object obj2) {
        return 42;
    }   
    public int compute(String s, Object... objects)
    {
        return 43;
    }

    public static void main(String[] args) {
        MethodResolutionTest2 mrt2 = new MethodResolutionTest2();
        System.out.println(mrt2.compute("hi",  mrt2));  
        System.out.println(mrt2.compute("hi",  new Object[]{mrt2}));    
        System.out.println(mrt2.compute("hi",  new Object[]{mrt2, mrt2, mrt2}));
    }
}
Run Code Online (Sandbox Code Playgroud)

打印出来的

42
43
43
Run Code Online (Sandbox Code Playgroud)

我理解第一行:JLS15.12表示方法解决是分阶段进行的,而第1阶段和第2阶段忽略了varargs方法,以确定是否存在兼容方法,仅当阶段1和阶段2失败时才会发生阶段3(包括变量).(参见JLS和这个问题.)compute(String s, Object... objects)如果compute(Object obj1, Object obj2)适用,总是被忽略.

但我不明白为什么43为其他两行打印.An Object[]也是一个实例Object,为什么它与varargs方法匹配?


编辑:

...还有这个

Object arg2 = new Object[]{mrt2};
System.out.println(mrt2.compute("hi", arg2));   
Run Code Online (Sandbox Code Playgroud)

打印42.

And*_*mas 9

在第8.4.1节中:

如果最后一个形式参数是类型的变量arity参数T,则认为它定义了类型的形式参数T[].

由于您明确提供了一个数组,因此允许后两个调用在第一阶段匹配变量arity方法,而不考虑变量arity.