Java 泛型陣列 不支援
阿新 • • 發佈:2019-02-15
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.
因為對於萬用字元的方式,最後取出資料是要做顯式的型別轉換的,所以並不會存在上一個例子的問題。