1. 程式人生 > >List與陣列相互轉換

List與陣列相互轉換

本篇文章介紹list與一維陣列之間的相互轉換關係:

List---->陣列:

方法一:List<String> list =new ArrayList<String>();
list.add("1");
list.add("2");
list.add("3");
list.add("4");
list.add("5");
list.add("6");
//String[] string =(String[]) list.toArray(new String[list.size()]);// 這裡可以這樣   相當於將list.toArray[list.size()]生成的陣列強制轉換給另一陣列(都是string型別的 所以可以轉換)

           String[] strings = new String[list.size()];

              list.toArray(strings);


for (String string2 : string) {
System.out.print(string2+",");
}

輸出結果:1,2,3,4,5,6,

方法二:

List<String> list =new ArrayList<String>();
list.add("1");
list.add("2");
list.add("3");
list.add("4");
list.add("5");
list.add("6");
String[] string =new String[list.size()];

                 for(int i=0;i<list.size();i++){

                  string[i]=list.get(i);

                          }
for (String string2 : string) {
System.out.print(string2+",");
}

輸出結果:1,2,3,4,5,6,


對於以下這種情況

List<String> list =new ArrayList<String>();

list.add("1");
list.add("2");
list.add("3");
list.add("4");
list.add("5");
list.add("6");
String[] string =(String[]) list.toArray();                 //toArray()中沒有引數
for (String string2 : string) {
System.out.print(string2+",");
}

輸出結果:Exception in thread "main" java.lang.ClassCastException: [Ljava.lang.Object; cannot be cast to [Ljava.lang.String;

at com.test.stringUtils.TTTTTTTTTTTTTTTTT.main(TTTTTTTTTTTTTTTTT.java:15)

對於這個現象我們可以這麼解釋:Java中允許向上和向下轉型,但是這個轉型是否成功是根據Java虛擬機器中這個物件的型別來實現的。Java虛擬機器中儲存了每個物件的型別。而陣列也是一個物件。陣列的型別是[Ljava.lang.Object。把[Ljava.lang.Object轉換成[Ljava.lang.String是顯然不可能的事情,因為這裡是一個向下轉型,而虛擬機器只儲存了這是一個Object的陣列,不能保證陣列中的元素是String的,所以這個轉型不能成功。數組裡面的元素只是元素的引用,不是儲存的具體元素,所以陣列中元素的型別還是儲存在Java虛擬機器中的。

根據上面的解釋,我們可以把這個問題歸納到下面這個模型:
Object objs[]=new Object[10];
String strs[]=(String[])objs;

這樣子和剛才上面編譯錯誤是一樣的。如果我們修改一下這個程式碼,如下:
String strs[]=new String[10];
Object objs[]=strs;

這樣子就可以編譯通過了。所以這個問題我們可以歸結為一個Java轉型規則的問題。下面談一下Java陣列對範型的支援問題。

JDK5中已經有了對範型的支援,這樣可以保證在集合和Map中的資料型別的安全,可是,List的toArray方法返回的竟然是Object []讓人很迷惑。個人感覺應該可以根據範型,直接返回相應的T []。仔細看了一下JDK的發現List轉化為array有兩個方法:

public Object[] toArray();
這個方法把List中的全部元素返回一個相同大小的陣列,陣列中的所有元素都為Object型別。

public <T> T[] toArray(T[] a);
這個方法把List中的全部元素返回一個相同大小的陣列,陣列中的所有元素都為T型別。

List如此設計是因為Java編譯器不允許我們new範型陣列。也就是說你不能這麼定義一個數組:
T arr=new T[size];

但是你卻可以用T[]來表示陣列,而且可以把陣列強制轉化為T[]。比如List中的public <T> T[] toArray(T[] a)是這麼實現的:
public <T> T[] toArray(T[] a) {
if (a.length < size)
a = (T[])java.lang.reflect.Array.
newInstance(a.getClass().getComponentType(), size);
System.arraycopy(elementData, 0, a, 0, size);
if (a.length > size)
a[size] = null;
return a;
}

從上面程式碼中可以看到,因為你不知道這個陣列的型別,你必須通過反射機制建立這個陣列(a.getClass().getComponentType()方法是取得一個數組元素的型別)。

最終,List轉換為Array可以這樣處理:

ArrayList<String> list=new ArrayList<String>();

String[] strings = new String[list.size()];

list.toArray(strings);

反過來,如果要將陣列轉成List怎麼辦呢?如下:

陣列--->轉換成list

String[] s = {"a","b","c"};
List list = java.util.Arrays.asList(s);

List<String> list =new ArrayList<String>();
list.add("1");
list.add("2");
list.add("3");
list.add("4");
list.add("5");
list.add("6");
String[] string =(String[]) list.toArray(new String[list.size()]);
for (String string2 : string) {
System.out.print(string2+",");
}

輸出結果:1,2,3,4,5,6,