- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
我意识到 Java 8 仍处于测试阶段,但这个让我觉得很奇怪:
public class Fields<C extends Enum<C>> {
public Fields(Set<C> columns) {
// A sample column used to find the universe of the enum of Columns.
C sampleCol = columns.iterator().next();
// Java 8 needs a cast here.
Set<C> allColumns = EnumSet.allOf((/*Class<C>)*/ sampleCol.getClass());
// ... there's more to this that I've deleted.
}
}
错误显示:
error: incompatible types: inferred type does not conform to equality constraint(s)
Set<C> allColumns = EnumSet.allOf(sampleCol.getClass());
inferred: C
equality constraints(s): C,CAP#1
where C is a type-variable:
C extends Enum<C> declared in class Test.Fields
where CAP#1 is a fresh type-variable:
CAP#1 extends Enum from capture of ? extends Enum
这是 Java 8 的错误还是新特性?
最佳答案
有趣,这是raw types 的处理方式的细微变化.
首先,让我们澄清一下您的示例。 Object.getClass
的返回类型很特别:
The actual result type is
Class<? extends |X|>
where|X|
is the erasure of the static type of the expression on whichgetClass
is called.
在这种情况下,X
将是类型参数 C
, 其中 erases至Enum
.所以sampleCol.getClass()
返回 Class<? extends Enum>
. EnumSet.allOf
声明类型参数E extends Enum<E>
, 在你的情况下 ? extends Enum
被推断为其类型参数。
重要的部分是 Enum
是原始类型。已经看到使用原始类型会删除看似无关的泛型,例如在这篇文章中:Why won't this generic java code compile?在他的回答中,Jon Skeet 引用了 JLS §4.8 (“原始类型”)来涵盖这种不直观的行为。
在您使用 Java 7 的示例中似乎发生了类似的行为:EnumSet.allOf(sampleCol.getClass())
允许编译时出现“未经检查的调用”警告(由于将生成的原始 EnumSet
分配给 Set<C>
,随后的“未经检查的转换”警告会隐藏此警告)。
问题变成了:在通用通配符的边界中出现的原始类型是否应该允许未经检查的转换? JLS §4.8 没有提到这一点,所以它是模棱两可的。可能这是一个错误,但这似乎是对这种行为的合理收紧。而像 Enum
这样的标准原始类型遗留 API 本身可能是预期的,一种“半生不熟”的类型,如 Class<? extends Enum>
只能发生在泛型之后,因此让它破坏泛型类型检查是没有意义的。
无论如何,我很想看看是否有人可以指出有关此更改的文档 - 我的搜索没有找到任何东西。
关于您的具体代码:您应该使用 getDeclaringClass()
反而。编译器无法知道调用 getClass
在 C
将准确返回 Class<C>
;事实上,如果在具有特定常量类的枚举上使用它就不会。这正是 Enum
的用例。声明该方法。
关于Java 8 需要强制转换,而 Java 7 不需要 - enum.getClass/getDeclaringClass,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21640002/
为什么这在 Java 中可行: this.getClass().getClass().getClass().getClass()... 为什么会出现这种无限递归? 只是好奇。 最佳答案 这里没有无限递
我正在测试Object是否等于特定的class类型。例如: @Override public void itemStateChanged(ItemEvent e) { if (e.getSta
详解java中this.getClass()和super.getClass()的实例 前言: 遇到this.getClass()和super.getClass()的返回值感到疑惑,经过探索豁然开
Blockquote public boolean equals(Object obj) { if (this == obj) { return true; } if (obj == nul
我正在尝试比较两个相同的对象。当页面第一次加载时,它们是相等的,但是当我第二次刷新页面时,尽管数据或对象没有改变,但它们并不相等。 我在做什么。 我有一个 CurrentUser obj,我在登录时将
我的 MyClass 类中有 equals() 的 @Override: @Entity( name = "MyClass" ) @Table( name = "my_class" ) public
我在这里真的是指身份平等。 例如,以下是否总是打印 true? System.out.println("foo".getClass() == "fum".getClass()); 最佳答案 是的,类标
getClass().getClassLoader().getResource()和getClass.getResource()有什么区别? 从资源中检索文件时,在什么情况下应该使用哪一个? 最佳答案
这个问题已经有答案了: Different ways of loading a file as an InputStream (6 个回答) 已关闭 5 年前。 我正在浏览 ImageIO,并且在网络
有没有办法创建另一个对象类型的新对象? 示例: Soldier extends Person Accountant extends Person Person 的每个子类都有一个接受(出生日期和死亡日
FXMLLoader类的 load()方法用于加载FXML文件。那么 getClass().getResource() 是在做什么 Parent root = FXMLLoader.load(getC
考虑这段代码: class A { static int i=3; } public class TT extends A { public static void main(Stri
为什么第一行有效而第二行无效: Class c1 = (new Object()).getClass().getClass(); Class> c2 = (new O
我见过类似的问题并且有一个可行的解决方案,但我没有深入理解为什么我的示例中的前四次尝试(c1、c2、c3 和 c4)无法编译。归结为我不理解 this.getClass() 返回的编译时类是什么。 i
这个 Actor 安全吗? private void foo(T value) { final Class aClass = (Class) value.getClass(); } 是否存在
我在学习 Java 时遇到了一件奇怪的事情。考虑以下程序: public class GetClassNameInheritance { public static void main(Str
我编写了以下代码: public class AnonymousClasses { public void sayHello(){ } public interface Greetin
我正在尝试创建其中包含文本字段的对话框。代码如下 private void showBatchDialog() { Dialog dialog = new Dialog(); dial
我正在尝试使用 getClass() 方法并具有以下代码: class parent{} class child extends parent{} public class test { pu
在 Java 中,我想为方法编写测试(简化片段): public class MyClass { private static final Set SOME_SET = new HashSet
我是一名优秀的程序员,十分优秀!