JDk---集合转数组的toArray()和toArray(T[] a)方法

来源:互联网 发布:组织架构及营销网络 编辑:程序博客网 时间:2024/06/05 16:01
代码是jdk ArrayList中的源码 

Java代码  收藏代码
  1.    
  2. public <T> T[] toArray(T[] a) {   
  3.         if (a.length < size)   
  4.             a = (T[])java.lang.reflect.Array.   
  5.                 newInstance(a.getClass().getComponentType(), size);   
  6.             System.arraycopy(elementData, 0, a, 0, size);   
  7.         if (a.length > size)   
  8.             a[size] = null;   
  9.         return a;   
  10.     }   
  11.   
  12.   
  13.   
  14.  public Object[] toArray()  
  15.     {  
  16.         Object aobj[] = new Object[size];  
  17.         System.arraycopy(((Object) (elementData)), 0, ((Object) (aobj)), 0, size);  
  18.         return aobj;  
  19.     }  
  20.   
  21.     public Object[] toArray(Object aobj[])  
  22.     {  
  23.         if(aobj.length < size)  
  24.             aobj = (Object[])(Object[])Array.newInstance(((Object) (aobj)).getClass().getComponentType(), size);  
  25.         System.arraycopy(((Object) (elementData)), 0, ((Object) (aobj)), 0, size);  
  26.         if(aobj.length > size)  
  27.             aobj[size] = null;  
  28.         return aobj;  
  29.     }  


1.该方法用了泛型,并且是用在方法的创建中(<T> 相当于定义泛型,T[]是在使用泛型T) 
泛型是Java SE 1.5的新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数。这种参数类型可以用在类、接口和方法的创建中,分别称为泛型类、泛型接口、泛型方法 
2.该方法返回集合中所有元素的数组;返回数组的运行时类型与指定数组的运行时类型相同。 


3.与 public Object[] toArray() 的比较 
public Object[] toArray() { 
Object[] result = new Object[size]; 
System.arraycopy(elementData, 0, result, 0, size); 
return result; 
    } 

从源码中可以看出它仅能返回 Object[]类型的,相当于toArray(new Object[0]) 注意:数组不能强制转换 

不带参数的toArray方法,是构造的一个Object数组,然后进行数据拷贝,此时进行转型就会产生ClassCastException 

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

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



带参数的toArray方法,则是根据参数数组的类型,构造了一个对应类型的,长度跟ArrayList的size一致的空数组,虽然方法本身还是以 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); 


1).参数指定空数组,节省空间 
String[] y = x.toArray(new String[0]); 
2).指定大数组参数浪费时间,采用反射机制 
String[] y = x.toArray(new String[100]);  //假设数组size大于100 
3).姑且认为最好的 
String[] y = x.toArray(new String[x.size()]); 


以下代码会出现ClassCastException 
List list = new ArrayList();   
list.add(new Long(1)); 
list.add(new Long(2));   
list.add(new Long(3)); 
list.add(new Long(4));   
Long[] l = (Long[])list.toArray();//这个语句会出现ClassCastException 


处理方式如下面代码: 
Long [] l = (Long []) list.toArray(new Long[list.size()]);   
0 0