Java 泛型數組

Java 不支持泛型數組。也就是說,

List<String>[] ls = new ArrayList<String>[10];  

是不支持的,而

List<String>[] ls = new ArrayList[10]  
卻可以。

是我一直不清楚爲什麼不能夠聲明泛型的數組,指定類型可以讓編譯的時候不會出現類型安全的提示。

直到今天我看到Sun的一篇文檔才清楚,裏面提到了一種情況:

List<String>[] lsa = new List<String>[10]; // Not really allowed.
Object o = lsa;
Object[] oa = (Object[]) o;
List<Integer> li = new ArrayList<Integer>();
li.add(new Integer(3));
oa[1] = li; // Unsound, but passes run time store check
String s = lsa[1].get(0); // Run-time error: ClassCastException.

這種情況下,由於JVM泛型的擦除機制,在運行時JVM是不知道泛型信息的,所以可以給oa[1]賦上一個ArrayList<Integer>而不會出現ArrayStoreException,但是在取出數據的時候卻要做一次類型轉換,所以就會出現ClassCastException,如果可以進行泛型數組的聲明,上面說的這種情況在編譯期將不會出現任何的警告和錯誤,只有在運行時纔會出錯。而對泛型數組的聲明進行限制,對於這樣的情況,可以在編譯期提示代碼有類型安全問題,比沒有任何提示要強很多。

基於以上的原因,Java不支持聲明泛型數組,更確切地表達是:數組的類型不可以是類型變量,除非是採用通配符的方式,看下面這個例子:

List<?>[] lsa = new List<?>[10]; // OK, array of unbounded wildcard type.
Object o = lsa;
Object[] oa = (Object[]) o;
List<Integer> li = new ArrayList<Integer>();
li.add(new Integer(3));
oa[1] = li; // Correct.
String s = (String) lsa[1].get(0); // Run time error, but cast is explicit.

因爲對於通配符的方式,最後取出數據是要做顯式的類型轉換的,所以並不會存在上一個例子的問題。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章