Java 5 和 6 在重载泛型方法时的不同行为
我在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中,我得到了预期的编译错误,指出“方法”的擦除是相同的:
$ 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)