- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
考虑以下 returnsNull
函数并使用泛型类型调用它:
public static <T> List<T> returnNull(Class<? extends T> clazz) {
return null;
}
public static void main( String[] args )
{
List<AtomicReference<?>> l = returnNull(AtomicReference.class);
}
Eclipse 编译器在设置为 Java 8 时接受它,但是 javac
在 Java 8 中拒绝它:
incompatible types: cannot infer type-variable(s) T
(argument mismatch; java.lang.Class<java.util.concurrent.atomic.AtomicReference> cannot be converted to java.lang.Class<? extends java.util.concurrent.atomic.AtomicReference<?>>)
潜在的区别似乎是给定两个参数化类型 P1<T>
和 P2<T>
,Eclipse 允许从使用原始内部类型参数化的外部类型进行转换:P1<P2>
到用内部类型的下界参数化的外部类型,带有无限通配符,如 P1<? extends P2<?>>
. javac
没有。
这不仅仅是理论上的思考:如果此代码被接受,它将解决我的 generics filtering problem .
谁是对的?
最佳答案
期间 applicability inference 欧洲法院推断<T>
至 AtomicReference#RAW
, 让我们签名 returnNull
显示为
List<AtomicReference#RAW> returnNull(Class<? extends AtomicReference#RAW>)
具体步骤是:
⟨Class<AtomicReference#RAW> → Class<? extends T#0>⟩
⟨Class<AtomicReference#RAW> <: Class<? extends T#0>⟩
⟨AtomicReference#RAW <= ? extends T#0⟩
⟨AtomicReference#RAW <: T#0⟩
AtomicReference#RAW <: T#0
T#0 = AtomicReference#RAW
现在,传递 Class<AtomicReference#RAW>
类型的值没有问题了进入那个方法。
(后缀 #RAW 是原始类型的特定于实现的表示,为清楚起见在此处复制)。
编辑: invocation type inference 期间情况看起来不同:通过将目标类型添加到混合中,我们最终(在 incorporation 期间)具有以下约束:
⟨AtomicReference#RAW <: AtomicReference<?>⟩
此约束应减少为 FALSE,但 ecj 减少为 TRUE。由此看来,拒绝该方案似乎是正确的答案。
我提交了 bug 528970以便欧洲法院进一步调查。
这有点讽刺意味,因为 javac has a long standing bug它错误地假设T#RAW <: T<X>
.出于兼容性原因,ECJ 明确地将此错误复制到许多位置,但显然在一个特定的代码位置是相反的:javac 在 ECJ 检查兼容性的地方应用子类型。
编辑 2: 一年后,ECJ 中的错误似乎只能在 JLS 在 JDK-8054721 及其周围得到改进后才能修复。 .
关于java - Eclipse ECJ 接受此代码,javac 不接受 - 谁是对的?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47662493/
我的问题是关于基本的java命令之一“javac”。我的桌面上有一个“.java”文件。我已经安装了 JDK 7 并将路径变量添加到环境变量中。这是“环境变量”的屏幕截图。 http://s13.po
我制作了我的 gitrepository 并提交了它。 插入了一个 java 文件并想编译它,但它给了我这个: Bernard@BERNARD-PC /c/users/bernard/desktop/
运行Maven时,得到以下输出: [WARNING] Unable to autodetect 'javac' path, using 'javac' from the environment. 我该
我有一个导入一些 servlet 库的类。当我从命令行编译它时它很好。 当我使用 ant compile 任务编译它时,它给出了在其路径中找不到 servlet 库的错误。 这是已知/常见的情况吗?
我试图在 maven-compiler-plugin 中指定另一个版本的 JDK .当-target和 -source参数设置为1.5,一切正常。但是当我尝试使用 1.6 JDK 时,maven 会报
我们的软件之前附带 OpenJDK JRE,但现在我们将附带 Oracle JRE。 之前我们使用 OpenJDK javac 编译器进行编译。我认为现在我们应该使用 Oracle javac 编译器
在大多数现代 IDE 中,您可以设置一个参数来确保 javac 获得足够的堆内存来进行编译。由于不值得在这里讨论的原因,我们暂时与 JBuilder 2005/2006 联系在一起,而且源代码的数量似
我在桌面上的 Notepad++ 中保存了一个名为“first.java”的文件。当我运行 cmd 命令“javac first.java”时,它给了我这个错误。 javac: file not fo
更新: See resolution here. 感谢大家的帮助! 我在尝试使用 Ant 编译项目时遇到错误,它声称“[javac] javac:无效目标版本:7”并导致构建失败。 我在 Mac OS
当我尝试在我的 gwt-maven Projekt 上进行 maven-install 时,我得到了这个错误: [ERROR] Failed to execute goal org.apache.ma
使用 maven 编译时出现编译错误。 [ERROR] COMPILATION ERROR : [INFO] ---------------------------------------------
我正在查看一些内部 javac sun 编译器 API 源代码,并在 Types 类中发现了这一点: public Boolean visitTypeVar(TypeVar var1, Type va
我正在尝试运行 java 应用程序,但出现以下错误, Unable to find a javac compiler; com.sun.tools.javac.Main is not on the c
我有这个类,它是我在从 Java 6 移植到 Java 8 的项目中找到的一些代码的简化: public class Unification { final class Box {}
首先,我要感谢你,并明确地说,我已经在这个问题上苦苦思索了好几天,并在其他类似线程中寻找解决方案,但没有成功。 我们的应用程序负责生成 java 类,其中一些可能在类名(因此文件名)中包含特殊字符,例
以下代码创建了一个Collector,它产生了一个UnmodifiableSortedSet: package com.stackoverflow; import java.util.Collecti
当我用 Maven 编译我的类时遇到问题。堆栈跟踪如下所示: [ERROR] Failure executing javac, but could not parse the error: [ERRO
这个问题在这里已经有了答案: Why Java compiler as distributed as executable and not as JVM bytecode? (1 个回答) 关闭 7
我有一些用 javac 1.8.0_92 编译的代码: public final class Either { // ... private final L l; privat
这个问题在这里已经有了答案: Lombok's access to jdk.compiler's internal packages incompatible with Java-16 (3 个回答)
我是一名优秀的程序员,十分优秀!