java中toArray()的用法

一直不太熟悉toarray()的用法 现整理如下

一.Incident

[java] view plain copy
 
  1. public static void main(String[] args) {  
  2.   
  3.     List<String> list = new ArrayList<String>();  
  4.   
  5.     list.add("1");  
  6.   
  7.     list.add("2");  
  8.   
  9.     String[] tt = (String[]) list.toArray(new String[0]);  
  10.   
  11. }  


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

二. Root Cause Analysis

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

[java] view plain copy
 
  1. public Object[] toArray() {  
  2.   
  3.     Object[] result = new Object[size];  
  4.   
  5.     System.arraycopy(elementData, 0, result, 0, size);  
  6.   
  7.     return result;  
  8.   
  9. }  

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

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

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

三. Solutions

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

[java] view plain copy
 
  1. 1. Long[] l = new Long[<total size>];  
  2.   
  3.    list.toArray(l);  
  4.   
  5. 2. Long[] l = (Long[]) list.toArray(new Long[0]);  
  6.   
  7. 3. Long[] a = new Long[<total size>];  
  8.   
  9.    Long[] l = (Long[]) list.toArray(a);  

 

四.Further Consideration

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

posted @ 2017-12-05 08:32  落落无伤  阅读(41583)  评论(0编辑  收藏  举报