Java中toArray的用法探究(java数组与list转换)

一.             Incident

import java.util.ArrayList;  

import java.util.List;  

 

public class Test {  

 

    public static void main(String[] args) {  

          

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

        list.add("1");   

        list.add("2");   

        String[] tt =(String[]) list.toArray(new String[0]);   

 

    }  

 

}

 

这段代码是没问题的,但我们看到String[] tt =(String[]) list.toArray(new String[0]) 中的参数很奇怪,然而去掉这个参数new String[0]却在运行时报错。。。

 

二.             Root Cause Analysis

经研究发现toArray有两个方法:

public Object[] toArray() {  

   Object[] result = new Object[size];     

   System.arraycopy(elementData, 0, result, 0, size);     

   return result;  

}  

不带参数的toArray方法,是构造的一个Object数组,然后进行数据拷贝,此时进行转型就会产生ClassCastException,这也就是上述问题的root cause了。

 

 

public Object[] toArray(Object a[]) {     

if (a.length < size)         

a = (Object[])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;  

}

而带参数的toArray方法,则是根据参数数组的类型,构造了一个对应类型的,长度跟ArrayListsize一致的空数组,虽然方法本身还是以 Object数组的形式返回结果,不过由于构造数组使用的ComponentType跟需要转型的ComponentType一致,就不会产生转型异常。

 

三.             Solutions

因此在使用toArray的时候可以参考以下三种方式

1. Long[] l = new Long[<total size>];

list.toArray(l);

 

      2. Long[] l = (Long[]) list.toArray(new Long[0]);

 

      3. Long[] a = new Long[<total size>];

      Long[] l = (Long[]) list.toArray(a);

 

四.Further Consideration

     该容器中的元素已经用泛型限制了,那里面的元素就应该被当作泛型类型的来看了,然而在目前的java中却不是的,当直接String[] tt =(String[]) list.toArray()时,运行报错。回想一下,应该是java中的强制类型转换只是针对单个对象的,想要偷懒,将整个数组转换成另外一种类型的数组是不行的,这和数组初始化时需要一个个来也是类似的。

 

以上From:    http://hi.baidu.com/%B4%CB%D6%D0%D3%D0%D5%E6%B5%C0/blog/item/8add93ec812dc9deb31cb1b0.html

 

 

 

[java]  view plain  copy
 
  1. import java.util.*;  
  2.   
  3. class testList{  
  4.     public static void main(String[] args){  
  5.         //List<Character> list = new ArrayList<Character>();  
  6.         String[] str = "a b c".split(" ");    
  7.         System.out.println("字符串数组str,长度为:" + str.length);  
  8.         for(String c : str)  
  9.             System.out.println(c);  
  10.               
  11.               
  12.         List<Integer> list = new ArrayList<Integer>();  
  13.         list.add(1);  
  14.         list.add(2);  
  15.         list.add(3);  
  16.         list.add(4);  
  17.         Iterator<Integer> it = list.iterator();  
  18.         System.out.println("整型List list");  
  19.         while(it.hasNext())  
  20.         {  
  21.             System.out.println(it.next());  
  22.         }  
  23.           
  24.         //数组转换成List  
  25.         List<String> li = new ArrayList<String>();  
  26.         li = Arrays.asList(str);         
  27.         Iterator<String> its = li.iterator();  
  28.         System.out.println("数组str -> list:");  
  29.         while(its.hasNext())  
  30.         {  
  31.             System.out.println(its.next());  
  32.         }  
  33.           
  34.         //List转换成数组  
  35.         Integer[] b = (Integer[])list.toArray(new Integer[0]);  
  36.         System.out.println("List list -> 数组b");  
  37.         for(Integer i : b)  
  38.             System.out.println(i);  
  39.           
  40.         //数组的fill方法  
  41.         int[] a = new int[5];  
  42.         Arrays.fill(a,0);  
  43.         System.out.println("数组a被0填充");  
  44.         for(int i : a)  
  45.             System.out.println(i);            
  46.     }  
  47. }  

下面的代码执行后:

 

[java]  view plain  copy
 
  1. private void onButtonListToArrayClick() {  
  2.     // Array--->List  
  3.     String[] strArray = new String[] { "aa""bb" };  
  4.     List<String> list = new ArrayList<String>(Arrays.asList(strArray));  
  5.     list.add("cc");  
  6.     // List--->Array  
  7.     String[] strArray2;  
  8.     try {  
  9.         strArray2 = (String[]) list.toArray();  
  10.     } catch (Exception e) {  
  11.         Log.i("ToArray""failure:" + e.getMessage());  
  12.     }  
  13.   
  14. }  


异常发生

 

原因为list.toArray()返回的是Object类型,强制转换的时候发生ClassCastException

程序修改如下:

 

[java]  view plain  copy
 
  1. private void onButtonListToArrayClick() {  
  2.     // Array--->List  
  3.     String[] strArray = new String[] { "aa""bb" };  
  4.     List<String> list = new ArrayList<String>(Arrays.asList(strArray));  
  5.     list.add("cc");  
  6.     // List--->Array  
  7.     String[] strArray2 = null;  
  8.     try {  
  9.         strArray2 = list.toArray(new String[0]);  
  10.     } catch (Exception e) {  
  11.         Log.i("ToArray""failure:" + e.getMessage());  
  12.     }  
  13.     for (String element : strArray2) {  
  14.         Log.i("ToArray""strArray2:" + element);  
  15.     }  
  16.   
  17. }  


运行结果:

 

上面的方法

 

[java]  view plain  copy
 
  1. strArray2 = list.toArray(new String[0]);  


与下面的方法一样

 

 

[java]  view plain  copy
 
  1. strArray2 = new String[list.size()];  
  2. strArray2 = list.toArray(strArray2);  


备注:

 

下面是ArrayList的两个toArray()方法的源代码:

 

[java]  view plain  copy
 
  1. public Object[] toArray() {   
  2.    Object[] result = new Object[size];     
  3.    System.arraycopy(elementData, 0, result, 0, size);     
  4.    return result;   
  5. }   
  6. public Object[] toArray(Object a[]) {     
  7. if (a.length < size)         
  8. a = (Object[])java.lang.reflect.Array.newInstance(a.getClass().getComponentType(), size);       System.arraycopy(elementData, 0, a, 0, size);     
  9. if (a.length > size)         
  10. a[size] = null;     
  11. return a;   
  12. }   


①不带参数的toArray方法,是构造的一个Object数组,然后进行数据拷贝,此时进行转型就会产生ClassCastException 
②带参数的toArray方法,则是根据参数数组的类型,构造了一个对应类型的,长度跟ArrayList的size一致的空数组,虽然方法本身还是以 Object数组的形式返回结果,不过由于构造数组使用的ComponentType跟需要转型的ComponentType一致,就不会产生转型异常

http://blog.csdn.net/gh102/article/details/7301051

http://blog.csdn.net/guopengzhang/article/details/5497875

猜你喜欢

转载自aoyouzi.iteye.com/blog/2284707