Java 10 个与 Collection 和 Generics 相关的迁移问题
Java 10 migration issue related to Collection and Generics
继续我之前在 link 中的查询:,我发现了更多问题(仅突出显示错误),这次我发现问题主要在集合 API 移动到 java 10.
之后
错误如下。想知道从 Java 8 迁移时 Java 10 是否有任何重大变化(从 Collection/Generics 的角度来看)。
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\ui\tree\WMTreeNode.java:159: error: breadthFirstEnumeration() in WMTreeNode cannot override breadthFirstEnumeration() in DefaultMutableTreeNode
[javac] public Enumeration<?extends WMTreeNode> breadthFirstEnumeration() {
[javac] ^
[javac] return type Enumeration<? extends WMTreeNode> is not compatible with Enumeration<TreeNode>
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\ui\tree\WMTreeNode.java:158: error: method does not override or implement a method from a supertype
[javac] @Override
[javac] ^
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\ui\tree\WMTreeNode.java:164: error: depthFirstEnumeration() in WMTreeNode cannot override depthFirstEnumeration() in DefaultMutableTreeNode
[javac] public Enumeration<?extends WMTreeNode> depthFirstEnumeration() {
[javac] ^
[javac] return type Enumeration<? extends WMTreeNode> is not compatible with Enumeration<TreeNode>
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\ui\tree\WMTreeNode.java:163: error: method does not override or implement a method from a supertype
[javac] @Override
[javac] ^
[javac] ^
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\ui\tree\WMTreeNode.java:298: error: no suitable method found for sort(Vector<TreeNode>)
[javac] Collections.sort(children);
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\datamgmt\ui\load\LoadNode.java:90: error: breadthFirstEnumeration() in LoadNode cannot override breadthFirstEnumeration() in DefaultMutableTreeNode
[javac] public Enumeration<?extends LoadNode> breadthFirstEnumeration() {
[javac] ^
[javac] return type Enumeration<? extends LoadNode> is not compatible with Enumeration<TreeNode>
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\datamgmt\ui\load\LoadNode.java:95: error: depthFirstEnumeration() in LoadNode cannot override depthFirstEnumeration() in DefaultMutableTreeNode
[javac] public Enumeration<?extends LoadNode> depthFirstEnumeration() {
[javac] where T is a type-variable:
[javac] T extends Object declared in class Class
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\automation\plot\CopyCountJobParameterDescriptor.java:113: error: incompatible types: Integer cannot be converted to CAP#1
[javac] final boolean outOfRange = (model.getMinimum().compareTo(i) > 0) || (model.getMaximum().compareTo(i) < 0);
[javac] ^
[javac] where CAP#1 is a fresh type-variable:
[javac] CAP#1 extends Object from capture of ?
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\automation\plot\CopyCountJobParameterDescriptor.java:113: error: incompatible types: Integer cannot be converted to CAP#1
[javac] final boolean outOfRange = (model.getMinimum().compareTo(i) > 0) || (model.getMaximum().compareTo(i) < 0);
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\datamgmt\ui\load\SnapshotLoadNode.java:48: error: breadthFirstEnumeration() in SnapshotLoadNode cannot override breadthFirstEnumeration() in DefaultMutableTreeNode
[javac] public Enumeration<?extends SnapshotLoadNode> breadthFirstEnumeration() {
[javac] ^
[javac] return type Enumeration<? extends SnapshotLoadNode> is not compatible with Enumeration<TreeNode>
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\datamgmt\ui\load\SnapshotLoadNode.java:53: error: depthFirstEnumeration() in SnapshotLoadNode cannot override depthFirstEnumeration() in DefaultMutableTreeNode
[javac] public Enumeration<?extends SnapshotLoadNode> depthFirstEnumeration() {
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\datamgmt\ui\favorites\FavoritesTreeNode.java:30: error: breadthFirstEnumeration() in FavoritesTreeNode cannot override breadthFirstEnumeration() in DefaultMutableTreeNode
[javac] public Enumeration<?extends FavoritesTreeNode> breadthFirstEnumeration() {
[javac] ^
[javac] return type Enumeration<? extends FavoritesTreeNode> is not compatible with Enumeration<TreeNode>
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\datamgmt\ui\favorites\FavoritesTreeNode.java:35: error: depthFirstEnumeration() in FavoritesTreeNode cannot override depthFirstEnumeration() in DefaultMutableTreeNode
[javac] public Enumeration<?extends FavoritesTreeNode> depthFirstEnumeration() {
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\datamgmt\ui\save\SaveNode.java:113: error: breadthFirstEnumeration() in SaveNode cannot override breadthFirstEnumeration() in DefaultMutableTreeNode
[javac] public Enumeration<?extends SaveNode> breadthFirstEnumeration() {
[javac] ^
[javac] return type Enumeration<? extends SaveNode> is not compatible with Enumeration<TreeNode>
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\datamgmt\ui\save\SaveNode.java:118: error: depthFirstEnumeration() in SaveNode cannot override depthFirstEnumeration() in DefaultMutableTreeNode
[javac] public Enumeration<?extends SaveNode> depthFirstEnumeration() {
[javac] ^
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\OSMWebServices\src\java\com\osm\webservices\legacy\util\MessageAttachmentHandler.java:76: error: cannot find symbol
[javac] attachment.setRawContent(new BufferedInputStream(new FileInputStream(file)),
[javac]
^
WMTreeNode
正在扩展 javax.swing.tree.DefaultMutableTreeNode
.
问题
如果仔细比较 DefaultMutableTreeNode
中 Java 8 和 10 之间的代码,您会发现 return 类型被泛化了:
// Java 8 uses `Enumeration` as a "raw type"
public Enumeration breadthFirstEnumeration() { /*...*/ }
// Java 10 uses `Enumeration` as intended: with a type parameter
public Enumeration<TreeNode> breadthFirstEnumeration() { /*...*/ }
你的代码是这样的:
public Enumeration<?extends SnapshotLoadNode> breadthFirstEnumeration() { /*...*/ }
您的代码在 Java 8 上编译,因为出于兼容性原因,可以将泛型添加到 raw type。虽然可能会有问题(实际上与下面相同),但无论如何原始类型都被认为是不安全的,所以这些都是可以预料的。
它不再在 Java 10 上编译,因为原来的 API 想要 return 一个 Enumeration<TreeNode>
,但你的覆盖不会那样做。相反,它 return 是一个 Enumeration
的子类型。现在,对于 Enumeration
这实际上不是问题,因为它是只读的(return 更具体的类型当然总是可以的),但编译器不知道(一个方法可以接受更具体的类型——那不行。
要更详细地理解最后一个括号,请假设它是 List<TreeNode>
和 List<SnapshotLoadNode>
。然后你的 API,用作 DefaultMutableTreeNode
,将 return 一个列表,呼叫者可以添加 TreeNode
-s 到。但是你的 class 仍然认为它有一个子类型列表并且会得到 class 转换异常 - 不好。如果没有原始类型,这种类型安全性的缺失是不可接受的,因此编译器会抱怨。
解决方法
如果 DefaultMutableTreeNode
是我的代码,我会将 breadthFirstEnumeration
更改为 return Enumeration<? extends TreeNode>
(我不知道为什么不是这样)。这将使您的代码编译。
由于这不是一个选项,所以看起来您必须这样做:
- 松开更精确的类型信息并将覆盖的 returns 类型更改为
Enumeration<TreeNode>
如果您需要更精确的 return 类型,请创建一个新方法:
public Enumeration<?extends SnapshotLoadNode>
breadthFirstEnumerationAsSnapshotLoadNode() { /*...*/ }
继续我之前在 link 中的查询:
错误如下。想知道从 Java 8 迁移时 Java 10 是否有任何重大变化(从 Collection/Generics 的角度来看)。
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\ui\tree\WMTreeNode.java:159: error: breadthFirstEnumeration() in WMTreeNode cannot override breadthFirstEnumeration() in DefaultMutableTreeNode
[javac] public Enumeration<?extends WMTreeNode> breadthFirstEnumeration() {
[javac] ^
[javac] return type Enumeration<? extends WMTreeNode> is not compatible with Enumeration<TreeNode>
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\ui\tree\WMTreeNode.java:158: error: method does not override or implement a method from a supertype
[javac] @Override
[javac] ^
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\ui\tree\WMTreeNode.java:164: error: depthFirstEnumeration() in WMTreeNode cannot override depthFirstEnumeration() in DefaultMutableTreeNode
[javac] public Enumeration<?extends WMTreeNode> depthFirstEnumeration() {
[javac] ^
[javac] return type Enumeration<? extends WMTreeNode> is not compatible with Enumeration<TreeNode>
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\ui\tree\WMTreeNode.java:163: error: method does not override or implement a method from a supertype
[javac] @Override
[javac] ^
[javac] ^
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\ui\tree\WMTreeNode.java:298: error: no suitable method found for sort(Vector<TreeNode>)
[javac] Collections.sort(children);
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\datamgmt\ui\load\LoadNode.java:90: error: breadthFirstEnumeration() in LoadNode cannot override breadthFirstEnumeration() in DefaultMutableTreeNode
[javac] public Enumeration<?extends LoadNode> breadthFirstEnumeration() {
[javac] ^
[javac] return type Enumeration<? extends LoadNode> is not compatible with Enumeration<TreeNode>
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\datamgmt\ui\load\LoadNode.java:95: error: depthFirstEnumeration() in LoadNode cannot override depthFirstEnumeration() in DefaultMutableTreeNode
[javac] public Enumeration<?extends LoadNode> depthFirstEnumeration() {
[javac] where T is a type-variable:
[javac] T extends Object declared in class Class
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\automation\plot\CopyCountJobParameterDescriptor.java:113: error: incompatible types: Integer cannot be converted to CAP#1
[javac] final boolean outOfRange = (model.getMinimum().compareTo(i) > 0) || (model.getMaximum().compareTo(i) < 0);
[javac] ^
[javac] where CAP#1 is a fresh type-variable:
[javac] CAP#1 extends Object from capture of ?
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\automation\plot\CopyCountJobParameterDescriptor.java:113: error: incompatible types: Integer cannot be converted to CAP#1
[javac] final boolean outOfRange = (model.getMinimum().compareTo(i) > 0) || (model.getMaximum().compareTo(i) < 0);
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\datamgmt\ui\load\SnapshotLoadNode.java:48: error: breadthFirstEnumeration() in SnapshotLoadNode cannot override breadthFirstEnumeration() in DefaultMutableTreeNode
[javac] public Enumeration<?extends SnapshotLoadNode> breadthFirstEnumeration() {
[javac] ^
[javac] return type Enumeration<? extends SnapshotLoadNode> is not compatible with Enumeration<TreeNode>
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\datamgmt\ui\load\SnapshotLoadNode.java:53: error: depthFirstEnumeration() in SnapshotLoadNode cannot override depthFirstEnumeration() in DefaultMutableTreeNode
[javac] public Enumeration<?extends SnapshotLoadNode> depthFirstEnumeration() {
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\datamgmt\ui\favorites\FavoritesTreeNode.java:30: error: breadthFirstEnumeration() in FavoritesTreeNode cannot override breadthFirstEnumeration() in DefaultMutableTreeNode
[javac] public Enumeration<?extends FavoritesTreeNode> breadthFirstEnumeration() {
[javac] ^
[javac] return type Enumeration<? extends FavoritesTreeNode> is not compatible with Enumeration<TreeNode>
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\datamgmt\ui\favorites\FavoritesTreeNode.java:35: error: depthFirstEnumeration() in FavoritesTreeNode cannot override depthFirstEnumeration() in DefaultMutableTreeNode
[javac] public Enumeration<?extends FavoritesTreeNode> depthFirstEnumeration() {
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\datamgmt\ui\save\SaveNode.java:113: error: breadthFirstEnumeration() in SaveNode cannot override breadthFirstEnumeration() in DefaultMutableTreeNode
[javac] public Enumeration<?extends SaveNode> breadthFirstEnumeration() {
[javac] ^
[javac] return type Enumeration<? extends SaveNode> is not compatible with Enumeration<TreeNode>
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\src\com\osm\datamgmt\ui\save\SaveNode.java:118: error: depthFirstEnumeration() in SaveNode cannot override depthFirstEnumeration() in DefaultMutableTreeNode
[javac] public Enumeration<?extends SaveNode> depthFirstEnumeration() {
[javac] ^
[javac] C:\WorkSpace\JAVA10\oswm\rel20.10_Patches\WorkManager\OSMWebServices\src\java\com\osm\webservices\legacy\util\MessageAttachmentHandler.java:76: error: cannot find symbol
[javac] attachment.setRawContent(new BufferedInputStream(new FileInputStream(file)),
[javac]
^
WMTreeNode
正在扩展 javax.swing.tree.DefaultMutableTreeNode
.
问题
如果仔细比较 DefaultMutableTreeNode
中 Java 8 和 10 之间的代码,您会发现 return 类型被泛化了:
// Java 8 uses `Enumeration` as a "raw type"
public Enumeration breadthFirstEnumeration() { /*...*/ }
// Java 10 uses `Enumeration` as intended: with a type parameter
public Enumeration<TreeNode> breadthFirstEnumeration() { /*...*/ }
你的代码是这样的:
public Enumeration<?extends SnapshotLoadNode> breadthFirstEnumeration() { /*...*/ }
您的代码在 Java 8 上编译,因为出于兼容性原因,可以将泛型添加到 raw type。虽然可能会有问题(实际上与下面相同),但无论如何原始类型都被认为是不安全的,所以这些都是可以预料的。
它不再在 Java 10 上编译,因为原来的 API 想要 return 一个 Enumeration<TreeNode>
,但你的覆盖不会那样做。相反,它 return 是一个 Enumeration
的子类型。现在,对于 Enumeration
这实际上不是问题,因为它是只读的(return 更具体的类型当然总是可以的),但编译器不知道(一个方法可以接受更具体的类型——那不行。
要更详细地理解最后一个括号,请假设它是 List<TreeNode>
和 List<SnapshotLoadNode>
。然后你的 API,用作 DefaultMutableTreeNode
,将 return 一个列表,呼叫者可以添加 TreeNode
-s 到。但是你的 class 仍然认为它有一个子类型列表并且会得到 class 转换异常 - 不好。如果没有原始类型,这种类型安全性的缺失是不可接受的,因此编译器会抱怨。
解决方法
如果 DefaultMutableTreeNode
是我的代码,我会将 breadthFirstEnumeration
更改为 return Enumeration<? extends TreeNode>
(我不知道为什么不是这样)。这将使您的代码编译。
由于这不是一个选项,所以看起来您必须这样做:
- 松开更精确的类型信息并将覆盖的 returns 类型更改为
Enumeration<TreeNode>
如果您需要更精确的 return 类型,请创建一个新方法:
public Enumeration<?extends SnapshotLoadNode> breadthFirstEnumerationAsSnapshotLoadNode() { /*...*/ }