在下面的课程中,由于对this()
的调用模糊,我在Java 8中遇到编译错误。但是,使用Java 6,这个类编译得很好。
我知道我可以使用工厂方法重构这个,但是对于发生问题的实际类,我强烈希望现在维护当前的API。
有人能想出一种方法来解决歧义而不改变外部API吗?
public class Vararg8 {
public Vararg8(final Object... os) {}
public Vararg8(final boolean b,
final String s,
final int... is) {}
public Vararg8() {
this(true, "test", 4, 5, 6);
}
}
答案 0 :(得分:18)
您可以通过传递明确的int[]
数组来执行此操作:
public Vararg8()
{
this(true, "test", new int[]{4, 5, 6});
}
您可能会注意到,从某种意义上说,这仍然是模棱两可的:您传递的内容仍然与Object...
构造函数兼容。这样做的原因是方法分辨率处于不同阶段,只有最后阶段允许考虑varargs参数。因为你已经使用了一个显式数组,所以它可以很好地击中第二个数组,而不需要扩展varargs。没有varargs扩展它就无法击中第一个,所以直到最后阶段都不会考虑。
第一阶段(§15.12.2.2)执行重载解析而不允许装箱或拆箱转换,或使用变量arity方法调用。如果在此阶段没有找到适用的方法,则处理继续到第二阶段。
第二阶段(§15.12.2.3)在允许装箱和拆箱的同时执行重载解析,但仍然排除使用变量arity方法调用。如果在此阶段没有找到适用的方法,则处理继续到第三阶段。
第三阶段(§15.12.2.4)允许重载与变量arity方法,装箱和拆箱相结合。
答案 1 :(得分:3)
试试这个:
public Vararg8()
{
this(true, "test", new int[]{4, 5, 6});
}
答案 2 :(得分:2)
使用显式int数组可以解决您的问题。
public Vararg8() {
this(true, "test",new int[]{ 4, 5, 6});
}
答案 3 :(得分:1)
您可以使用generic在运行时推断类型,但这会将它们转换为Boxed形式。这当然是性能神风,如果你在那里做很多算术或者必须打包很多原语,但会让所有现有的代码都正常工作。
如果您已经使用了对象类型,那么此解决方案将不会花费任何费用。
这看起来像这样:
public<A extends Boolean, B extends String, C extends Integer> Disambiguate(final A booleanPar,
final B stringPar,
final C... integerPar) {System.out.println("Im in the specific one");}
public<T extends Object> Disambiguate(final T... os) {System.out.println("Im in the general one");}
public static void main(String[] args) {
new Disambiguate(true, "test", 4, 5, 6);
}
您可以将泛型用于1.5及更高版本的“向后兼容性”,并使所有现有代码正常工作并制作新的Api,以避免将来出现此问题。
答案 4 :(得分:0)
字符数组也可以解决您的问题
public Vararg8() {
this(true, "test".toCharArray(), 4, 5, 6);
}