本文實(shí)例分析了Java中List與數(shù)組相互轉(zhuǎn)換的方法。分享給大家供大家參考。具體如下:
今天寫代碼遇到一個(gè)奇怪的問(wèn)題,具體代碼不貼出了,寫一個(gè)簡(jiǎn)化的版本。如下:
ArrayList<String> list=new ArrayList<String>();String strings[]=(String [])list.toArray();
這樣寫代碼個(gè)人覺(jué)得應(yīng)該沒(méi)什么問(wèn)題,編譯也沒(méi)有問(wèn)題。可是具體運(yùn)行的時(shí)候報(bào)異常,如下:Exception in thread "main" java.lang.ClassCastException: [Ljava.lang.Object;
但是這么寫是沒(méi)有問(wèn)題的:
ArrayList<String> list=new ArrayList<String>();String strings[]=new String[list.size()];for(int i=0,j=list.size();i<j;i++){ strings[i]=list.get(i);}
對(duì)于這個(gè)現(xiàn)象我們可以這么解釋:Java中允許向上和向下轉(zhuǎn)型,但是這個(gè)轉(zhuǎn)型是否成功是根據(jù)Java虛擬機(jī)中這個(gè)對(duì)象的類型來(lái)實(shí)現(xiàn)的。Java虛擬機(jī)中保存了每個(gè)對(duì)象的類型。而數(shù)組也是一個(gè)對(duì)象。數(shù)組的類型[Ljava.lang.Object。把[Ljava.lang.Object轉(zhuǎn)換成[Ljava.lang.String是顯然不可能的事情,因?yàn)檫@里是一個(gè)向下轉(zhuǎn)型,而虛擬機(jī)只保存了這是一個(gè)Object的數(shù)組,不能保證數(shù)組中的元素是String的,所以這個(gè)轉(zhuǎn)型不能成功。數(shù)組里面的元素只是元素的引用,不是存儲(chǔ)的具體元素,所以數(shù)組中元素的類型還是保存在Java虛擬機(jī)中的。
根據(jù)上面的解釋,我們可以把這個(gè)問(wèn)題歸納到下面這個(gè)模型:
Object objs[]=new Object[10];String strs[]=(String[])objs;
這樣子和剛才上面編譯錯(cuò)誤是一樣的。如果我們修改一下這個(gè)代碼,如下:
String strs[]=new String[10];Object objs[]=strs;
這樣子就可以編譯通過(guò)了。所以這個(gè)問(wèn)題我們可以歸結(jié)為一個(gè)Java轉(zhuǎn)型規(guī)則的問(wèn)題。下面談一下Java數(shù)組對(duì)范型的支持問(wèn)題。
JDK5中已經(jīng)有了對(duì)范型的支持,這樣可以保證在集合和Map中的數(shù)據(jù)類型的安全,可是,List的toArray方法返回的竟然是Object []讓人很迷惑。個(gè)人感覺(jué)應(yīng)該可以根據(jù)范型,直接返回相應(yīng)的T []。仔細(xì)看了一下JDK的源碼發(fā)現(xiàn)List轉(zhuǎn)化為array有兩個(gè)方法:
public Object[] toArray();
這個(gè)方法把List中的全部元素返回一個(gè)相同大小的數(shù)組,數(shù)組中的所有元素都為Object類型。
public <T> T[] toArray(T[] a);
這個(gè)方法把List中的全部元素返回一個(gè)相同大小的數(shù)組,數(shù)組中的所有元素都為T類型。
List如此設(shè)計(jì)是因?yàn)閖ava編譯器不允許我們new范型數(shù)組。也就是說(shuō)你不能這么定義一個(gè)數(shù)組:
T arr=new T[size];
但是你卻可以用T[]來(lái)表示數(shù)組,而且可以把數(shù)組強(qiáng)制轉(zhuǎn)化為T[]。比如List中的public <T> T[] toArray(T[] a)是這么實(shí)現(xiàn)的:
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;}
從上面代碼中可以看到,因?yàn)槟悴恢肋@個(gè)數(shù)組的類型,你必須通過(guò)反射機(jī)制創(chuàng)建這個(gè)數(shù)組(a.getClass().getComponentType()方法是取得一個(gè)數(shù)組元素的類型)。
最終,List轉(zhuǎn)換為Array可以這樣處理:
ArrayList<String> list=new ArrayList<String>();String[] strings = new String[list.size()];list.toArray(strings);
反過(guò)來(lái),如果要將數(shù)組轉(zhuǎn)成List怎么辦呢?如下:
String[] s = {"a","b","c"};List list = java.util.Arrays.asList(s);
希望本文所述對(duì)大家的java程序設(shè)計(jì)有所幫助。
新聞熱點(diǎn)
疑難解答
圖片精選