java - 帮助 Java 泛型 : Cannot use "Object" as argument for "? extends Object"

标签 java generics compiler-errors

我有以下代码:

import java.util.*;

public class SellTransaction extends Transaction {
    private Map<String,? extends Object> origValueMap;
    public SellTransaction(Map<String,? extends Object> valueMap) {
        super(Transaction.Type.Sell);
        assignValues(valueMap);
        this.origValueMap=valueMap;
    }
    public SellTransaction[] splitTransaction(double splitAtQuantity) {
        Map<String,? extends Object> valueMapPart1=origValueMap;
        valueMapPart1.put(nameMappings[3],(Object)new Double(splitAtQuantity));
        Map<String,? extends Object> valueMapPart2=origValueMap;
        valueMapPart2.put(nameMappings[3],((Double)origValueMap.get(nameMappings[3]))-splitAtQuantity);
        return new SellTransaction[] {new SellTransaction(valueMapPart1),new SellTransaction(valueMapPart2)};
    }
}

当我调用 valueMapPart1.put 时代码编译失败和 valueMapPart2.put ,错误:

The method put(String, capture#5-of ? extends Object) in the type Map is not applicable for the arguments (String, Object)

我在网上看过关于泛型和通配符和捕获的内容,但我还是不明白哪里出了问题。我的理解是 Map 的值可以是任何扩展 Object 的类,我认为这可能是多余的,因为所有类都扩展了 Object。而且我不能将泛型更改为类似 ? super Object 的东西,因为 Map由一些图书馆提供。

那么为什么不能编译呢?另外,如果我尝试投 valueMapMap<String,Object> ,编译器会给我“未经检查的转换”警告。

谢谢!

最佳答案

如果库指定extends那么他们明确不允许 put .您应该在修改之前进行防御性复制,因为它们可以非常合法地将其返回类型更改为在新版本中不可变。如果复制成本很高,那么您可以尝试创建类型为 <String, Object> 的 map 类型首先查询他们的 map ,然后查询您创建的具有本地修改的 map 。

如果您确实知道它们的返回类型是不可变的并且您完全拥有它,那么 @SuppressWarnings("unchecked")注释是解决警告的合法方法,但我会仔细检查这些假设并广泛评论。

了解extends对比super ,这样看。 由于该值可以是扩展 Object 的任何类型, 以下是有效的。

Map<String, Number> strToNum = new HashMap<String, Number>();
strToNum.put("one", Integer.valueOf(1));  // OK

Map<String, String> strToStr = new HashMap<String, String>();
strToStr.put("one", "1");  // OK

Map<String, ? extends Object> strToUnk = randomBoolean() ? strToNum : strToStr;
strToUnk.put("null", null);  // OK.  null is an instance of every reference type.
strToUnk.put("two", Integer.valueOf(2));  // NOT OK.  strToUnk might be a string to string map
strToUnk.put("two", "2");  // NOT OK.  strToUnk might be a string to number map

所以 put不适用于 extends边界类型。 但它非常适合像 get 这样的读取操作:

Object value = strToUnk.get("one");  // We don't know whether value is Integer or String, but it is an object (or null).

如果您希望 map 主要使用“put”而不是“get”,那么您可以使用“super”而不是 extends,如下所示:

Map<String, Number> strToNum = new HashMap<String, Number>();
Map<String, Object> strToObj = new HashMap<String, Object>();

Map<String, ? super Number> strToNumBase;
if (randomBoolean()) {
  strToNumBase = strToNum;
} else {
  strToNumBase = strToObj;
}

// OK.  We know that any subclass of Number can be used as values.
strToNumBase.put("two", Double.valueOf(2.0d));

// But now, gets don't work as well.
Number n = strToNumBase.get("one");  // NOT OK. 

关于java - 帮助 Java 泛型 : Cannot use "Object" as argument for "? extends Object",我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4543592/

相关文章:

java - 在 macOS (Mac OS X) 上打印 Java 类路径

C# 是否可以创建可选的泛型类型约束

java - 有界泛型方法未编译 - 为什么?

c++ - Boost信号2未编译

c++ - 带有 cout 的编译器错误消息

java - 在 Spring Web Flux 中捕获异常

java - 缺少约束 : Import-Package: Not able to start Activator

scala - 是什么在模式保护中导致匹配元组列表的类型错误

java - 使用 JodaTime 对每个时间间隔进行计数

c# - 我应该如何清除通用静态类中的字段?