Я пытаюсь понять, почему этот код имеет предупреждение без предупреждения. Первые два броска не имеют предупреждения, но третий делает:
class StringMap<V> extends HashMap<String, V> {
}
class StringToIntegerMap extends HashMap<String, Integer> {
}
Map<?, ?> map1 = new StringToIntegerMap();
if (map1 instanceof StringToIntegerMap) {
StringToIntegerMap stringMap1 = (StringToIntegerMap)map1; //no unchecked cast warning
}
Map<String, Integer> map2 = new StringMap<>();
if (map2 instanceof StringMap) {
StringMap<Integer> stringMap2 = (StringMap<Integer>)map2; //no unchecked cast warning
}
Map<?, Integer> map3 = new StringMap<>();
if (map3 instanceof StringMap) {
StringMap<Integer> stringMap3 = (StringMap<Integer>)map3; //unchecked cast warning
}
Это полное предупреждение для stringMap3 cast:
Тип безопасности: снят флажок с
Map<capture#3-of ?,Integer>доStringMap<Integer>
Однако объявление класса StringMap указывает параметр первого типа Map (т.е. String), и оба map3 и приведение StringMap<Integer> используют один и тот же тип для параметра второго типа Map (т.е. Integer). Из того, что я понимаю, до тех пор, пока актер не бросает ClassCastException (и он не должен, так как существует проверка instanceof), stringMap3 будет действительным Map<String, Integer>.
Является ли это ограничением компилятора Java? Или есть сценарий, когда вызовы методов map3 или stringMap3 с определенными аргументами могут привести к неожиданному ClassCastException, если предупреждение игнорируется?