我有以下代码:
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
由一些图书馆提供。
那么为什么不能编译呢?另外,如果我尝试投 valueMap
至 Map<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/