zoukankan      html  css  js  c++  java
  • toArray方法-转载

    toArray方法可能报空指针异常和类型转换异常

    篇一:

    ArrayList提供了一个将List转为数组的一个非常方便的方法toArray。toArray有两个重载的方法:

    1.list.toArray();

    2.list.toArray(T[]  a);

    对于第一个重载方法,是将list直接转为Object[] 数组;

    第二种方法是将list转化为你所需要类型的数组,当然我们用的时候会转化为与list内容相同的类型。

    不明真像的同学喜欢用第一个,是这样写:

    1
    2
    3
    4
    5
    6
    7
    ArrayList<String> list=new ArrayList<String>();
            for (int i = 0; i < 10; i++) {
                list.add(""+i);
            }
             
            String[] array= (String[]) list.toArray();
            

    结果一运行,报错:

    Exception in thread "main" java.lang.ClassCastException: [Ljava.lang.Object; cannot be cast to [Ljava.lang.String;

    原因一看就知道了,不能将Object[] 转化为String[].转化的话只能是取出每一个元素再转化,像这样:

    1
    2
    3
    4
    5
    Object[] arr = list.toArray();
            for (int i = 0; i < arr.length; i++) {
                String e = (String) arr[i];
                System.out.println(e);
            }

    所以第一个重构方法就不是那么好使了。

    实际上,将list世界转化为array的时候,第二种重构方法更方便,用法如下:

    1
    2
    String[] array =new String[list.size()];
            list.toArray(array);

    篇二:

    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中的强制类型转换只是针对单个对象的,想要偷懒,将整个数组转换成另外一种类型的数组是不行的,这和数组初始化时需要一个个来也是类似的。

  • 相关阅读:
    C++对象模型
    Session、Application、Cache
    JavaScript事件的几个细节
    algorithm(算法)
    【REST WCF】30分钟理论到实践
    实践Scrum
    调试PostSharp DEMO 遇到的问题
    6.8 按字符串中的部分内容排序
    QObject就有eventFilter,功能很强(随心所欲的进行处理,比如用来QLineEdit分词)
    QPainter就是手里的作图工具,只需要三洋东西:笔(颜色,宽度,样式),字体(写字),刷子(大面积作画),这里有三个典型例子
  • 原文地址:https://www.cnblogs.com/lonely-buffoon/p/5574554.html
Copyright © 2011-2022 走看看