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

代码是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()]);   

你可能感兴趣的:(java,java.util.list)