我在Java的泛型中遇到了一个问题,同样的代码在Java 6中可以编译并正常工作,但由于相同的类型擦除,在Java 5中无法编译。我有一个名为TestErasure.java的文件,其中有一个重载的方法,称为"method":
import java.util.ArrayList;
import java.util.List;
public class TestErasure {
public static Object method(List<Object> list) {
System.out.println("method(List<Object> list)");
return null;
}
public static String method(List<String> list) {
System.out.println("method(List<String> list)");
return null;
}
public static void main(String[] args) {
method(new ArrayList<Object>());
method(new ArrayList<String>());
}
}
在Java 5中,我得到了预期的编译错误,指出"method"的擦除是相同的:
$ javac -version
javac 1.5.0_19
$ javac TestErasure.java
TestErasure.java:10: name clash: method(java.util.List<java.lang.String>) and method(java.util.List<java.lang.Object>) have the same erasure
public static String method(List<String> list) {
^
TestErasure.java:17: method(java.util.List<java.lang.Object>) in TestErasure cannot be applied to (java.util.ArrayList<java.lang.String>)
method(new ArrayList<String>());
^
2 errors
然而,Java 6 能够编译和运行相同的代码。
$ javac -version
javac 1.6.0_16
$ javac TestErasure.java
$ java TestErasure
method(List<Object> list)
method(List<String> list)
根据我对擦除的理解(感谢Jon Skeet和Angelika Langer),我实际上预期Java 5抛出的编译错误(除非Java处理泛型的方式发生了变化-我在Java 6发布说明中找不到相关信息)。事实上,如果我修改其中一个重载方法的返回类型:
public static Object method(List<Object> list) ...
public static Object method(List<String> list) ...
Java 6由于相同的类型擦除问题也无法编译通过:
$ javac TestErasure.java TestErasure.java:5: name clash: method(java.util.List<java.lang.Object>) and method(java.util.List<java.lang.String>) have the same erasure
public static Object method(List<Object> list) {
^
TestErasure.java:10: name clash: method(java.util.List<java.lang.String>) and method(java.util.List<java.lang.Object>) have the same erasure
public static Object method(List<String> list) {
^
2 errors
似乎在Java 6中,返回类型会以某种方式影响使用哪个重载方法?
有人能解释一下为什么第一个例子在Java 6中有效吗?它似乎违反了重载泛型方法的处理规则?
更多信息:
根据David的建议,原始示例由javac 1.6编译,将在java 1.5下运行:
$ javac -target 1.5 TestErasure.java
$ java -version
java version "1.5.0_19"
$ java TestErasure
method(List<Object> list)
method(List<String> list)