LinkedList<Integer> adjMatrix[] = new LinkedList[10];
Object[] temp = (Object[]) adjMatrix;
LinkedList<String> string = new LinkedList<String>();
string.add("abc");
temp[3] = string;
System.out.println(adjMatrix[3].get(0));
//prints abc
//====================================================
LinkedList<String>[] stringList = new LinkedList[10];
Object[] oa = (Object[]) stringList;
LinkedList<Integer> listInt = new LinkedList<Integer>();
listInt.add(new Integer(3));
oa[1] = listInt;
System.out.println(stringList[1].get(0));
// gives ClassCastException java.lang.Integer cannot be cast to java.lang.String
我很困惑第一种情况下究竟发生了什么,一个整数接受了一个字符串值,但它并没有在第二种情况下发生。有什么想法吗?
答案 0 :(得分:6)
在第一种情况下,您只是打印adjMatrix[3].get(0)
。编译器认为此表达式是Integer。但实际类型并不重要:生成的字节码是
37: invokevirtual #8 // Method java/util/LinkedList.get:(I)Ljava/lang/Object;
40: invokevirtual #9 // Method java/io/PrintStream.println:(Ljava/lang/Object;)V
所以它只是将它从列表中获取的对象传递给PrintStream.println(Object)
,然后在此对象上调用String.value()
。编译器从不向字节码添加强制转换,因为它不是必需的。
在第二种情况下,您打印stringList[1].get(0)+""
,编译器认为它是String与空String的串联。所以生成的字节码是
101: invokevirtual #8 // Method java/util/LinkedList.get:(I)Ljava/lang/Object;
104: checkcast #14 // class java/lang/String
107: invokevirtual #15 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
110: ldc #16 // String
112: invokevirtual #15 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
115: invokevirtual #17 // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
118: invokevirtual #18 // Method java/io/PrintStream.println:(Ljava/lang/String;)V
如您所见,这里有一个checkcast指令,因为字节码需要调用StringBuilder.append(String)
,因此需要将表达式强制转换为String。因为它实际上是一个Integer,所以它会引发ClassCastException。
答案 1 :(得分:2)
第一个示例创建一个损坏的数组。这就是为什么在尝试使用泛型数组时会收到警告的原因,因为JVM无法强制实现类型安全。
请记住,由于类型擦除,LinkedList<Integer> adjMatrix[]
在运行时实际上是LinkedList adjMatrix[]
,即它可以存储所有对象类型。