Java 5 和 6 在重载泛型方法时的不同行为

2022-09-02 22:30:48

我在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 SkeetAngelika 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)

答案 1

在Sun上发现了这些错误,我认为这就是你所描述的:


http://bugs.sun.com/view_bug.do?bug_id=6182950 http://bugs.sun.com/view_bug.do?bug_id=6730568


答案 2

推荐