如果图书馆指定extends
然后他们明确禁止put
。您应该在修改之前进行防御性复制,因为它们可以完全合法地将其返回类型更改为在新版本中不可变。如果复制成本很高,那么您可以尝试创建一个类型为<String, Object>
首先查询他们的地图,然后查询您创建的包含您本地修改的地图。
如果您确实知道它们的返回类型是不可变的并且您单独拥有它,那么@SuppressWarnings("unchecked")
注释是解决警告的合法方法,但我会仔细检查这些假设并进行广泛评论。
要了解extends
vs 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
So 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).
如果您希望地图主要使用“put”而不是“get”,那么您可以使用“super”而不是扩展,如下所示:
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.