Chinaunix首页 | 论坛 | 博客
  • 博客访问: 141294
  • 博文数量: 31
  • 博客积分: 1952
  • 博客等级: 上尉
  • 技术积分: 305
  • 用 户 组: 普通用户
  • 注册时间: 2010-03-24 10:56
文章分类

全部博文(31)

文章存档

2011年(2)

2010年(29)

最近访客

分类: Java

2010-08-22 18:38:57

一.             Incident

import java.util.ArrayList;  

import java.util.List;  

public class Test {  

    public static void main(String[] args) {  

          

        List list = new ArrayList();   

        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[];

list.toArray(l);

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

      3. Long[] a = new Long[];

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

四.Further Consideration

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