zoukankan      html  css  js  c++  java
  • (转)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方法,则是根据参数数组的类型,构造了一个对应类型的,长度跟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);

    四.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

  • 相关阅读:
    windows64系统下安装 redis服务 (详细)
    周期信号的傅里叶级数表示
    LeetCode 36——有效的数独
    LeetCode 3——无重复字符的最长子串
    线性时不变系统的卷积
    信号与系统
    C++ 学习笔记之——输入和输出
    LeetCode 74——搜索二维矩阵
    LeetCode 389——找不同
    LeetCode 2——两数相加
  • 原文地址:https://www.cnblogs.com/redlight/p/3001662.html
Copyright © 2011-2022 走看看