java - 在 Java 中转换为泛型类型不会引发 ClassCastException?

标签 java generics casting

我遇到了一个奇怪的 Java 行为,它看起来像是一个错误。是吗?将 Object 转换为通用类型(例如,K)不会抛出 ClassCastException,即使对象不是 K 的实例也是如此。这是一个例子:

import java.util.*;
public final class Test {
  private static<K,V> void addToMap(Map<K,V> map, Object ... vals) {
    for(int i = 0; i < vals.length; i += 2)
      map.put((K)vals[i], (V)vals[i+1]); //Never throws ClassCastException!
  }
  public static void main(String[] args) {
    Map<String,Integer> m = new HashMap<String,Integer>();
    addToMap(m, "hello", "world"); //No exception
    System.out.println(m.get("hello")); //Prints "world", which is NOT an Integer!!
  }
}

更新:感谢 cletus 和 Andrzej Doyle 提供的有用答案。因为我只能接受一个,所以我接受 Andrzej Doyle's answer因为它让我找到了一个我认为还不错的解决方案。我认为这是在单行代码中初始化小 map 的一种更好的方法。

  /**
   * Creates a map with given keys/values.
   * 
   * @param keysVals Must be a list of alternating key, value, key, value, etc.
   * @throws ClassCastException if provided keys/values are not the proper class.
   * @throws IllegalArgumentException if keysVals has odd length (more keys than values).
   */
  public static<K,V> Map<K,V> build(Class<K> keyClass, Class<V> valClass, Object ... keysVals)
  {
    if(keysVals.length % 2 != 0)
      throw new IllegalArgumentException("Number of keys is greater than number of values.");

    Map<K,V> map = new HashMap<K,V>();
    for(int i = 0; i < keysVals.length; i += 2)
      map.put(keyClass.cast(keysVals[i]), valClass.cast(keysVals[i+1]));

    return map;
  }

然后你这样调用它:

Map<String,Number> m = MapBuilder.build(String.class, Number.class, "L", 11, "W", 17, "H", 0.001);

最佳答案

Java 泛型使用类型删除,这意味着那些参数化类型不会在运行时保留,因此这是完全合法的:

List<String> list = new ArrayList<String>();
list.put("abcd");
List<Integer> list2 = (List<Integer>)list;
list2.add(3);

因为编译后的字节码看起来更像这样:

List list = new ArrayList();
list.put("abcd");
List list2 = list;
list2.add(3); // auto-boxed to new Integer(3)

Java 泛型只是转换 Object 的语法糖。

关于java - 在 Java 中转换为泛型类型不会引发 ClassCastException?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2767212/

相关文章:

java - 注释类型元素的语法是什么?

generics - 我可以避免使用泛型对特征实现进行急切的歧义解决吗?

java - 将类映射到参数与类的参数相同的泛型

java - 需要 Java 中类型转换的解释。哪种方式最好?

java - 字符串缓冲区 - OutOfMemoryError

java - 使用多线程从多个文件读取并将它们写入文件文件序列

swift - Swift 中是否可以自定义合成条件一致性?

c++ - 在 C++ 中转换函数指针的奇怪行为

c# - WCF 服务应该返回普通的旧对象,还是您正在使用的实际类?

java - 在java中定义大量变量