Java Bug--官方编号为 6260652

在看ArrayList源码时,发现了这样一行代码

 c.toArray might (incorrectly) not return Object[] (see 6260652)

意思是c.toArray()不一定会返回 Object[] ,请查看编号为6260652的Java Bug

这里附上Java Bug 网址: Java Bug Database,可以根据关键词或bug id 查询详细信息

看一组实例吧...

public class JavaBug {
    public static void test1() {
        System.out.println("this is test1...");
        List<String> list = new ArrayList<>();
        list.add("aa");
        list.add("bb");
        Object[] arr = list.toArray();
        System.out.println(arr.getClass().getCanonicalName());
        arr[0] = new Object();
        JavaBug.printArr(arr);
    }

    public static void test2() {
        System.out.println("this is test2...");
        List<String> list = Arrays.asList("aa", "bb");
        Object[] arr = list.toArray();
//        Object[] arr = list.toArray(new Object[0]); 正确代码,将数组转换为Object类型
        System.out.println(arr.getClass().getCanonicalName());
        arr[0] = new Object();
        JavaBug.printArr(arr);
    }

    public static void test3() {
        System.out.println("this is test3...");
        Object[] arr = new String[]{"aa", "bb"};
        arr[0] = 1;
        JavaBug.printArr(arr);
    }

    public static void printArr(Object[] arr) {
        for (Object o : arr) {
            System.out.print(o + "  ");
        }
        System.out.println();
    }

    public static void main(String[] args) {
        /* 分别依次调用以下三个函数*/
        /*
         * test1 Output:
         * this is test1...
         * java.lang.Object[]
         * java.lang.Object@2503dbd3  bb
         */
        JavaBug.test1();
        /*
         * test2 Output:
         * this is test2...
         * java.lang.String[]
         * Exception in thread "main" java.lang.ArrayStoreException: java.lang.Object
         */
        JavaBug.test2();
        /*
         * test3 Output:
         * this is test3...
         * Exception in thread "main" java.lang.ArrayStoreException: java.lang.Integer
         */
        JavaBug.test3();
    }
}

test1(),test2()都是调用toArray(),为什么一个更改arr[0]没问题,另一个报错呢?

首先看ArrayList.java中toArray()源码:由于ArrayList中elementData类型为Object[],所以调用copyOf()返回值类型为Object[]

public Object[] toArray() {
        return Arrays.copyOf(elementData, size);
    }

public static <T> T[] copyOf(T[] original, int newLength) {
        return (T[]) copyOf(original, newLength, original.getClass());
    }

 

再看一下Arrays.asList()源码及相关代码: asList()将String[]作为参数传入,然后调用了ArrayList的代参构造函数,此时a的真实类型依然是String[],然后调用clone(),a的类型依然是String[],所以最终调用了toArray()后,返回的是String[],这也就是为什么再讲数组第一个赋为new Object()时,因为类型不同所以不能进行转换,这与test3()的例子相似

public static <T> List<T> asList(T... a) {
        return new ArrayList<>(a);
    }

ArrayList(E[] array) {
            a = Objects.requireNonNull(array);
        }


public static <T> T requireNonNull(T obj) {
        if (obj == null)
            throw new NullPointerException();
        return obj;
    }

public Object[] toArray() {
            return a.clone();
        } 

总结: 

由于Arrays.asList()返回的不一定是Object[],所以会出现向数组中添加Object对象报异常的问题,问题是在2005年提出的,现在已经解决了,所以尽量使用toArray(T[] a)避免Exception的发生

public <T> T[] toArray(T[] a) {
        if (a.length < size)
            // Make a new array of a's runtime type, but my contents:
            return (T[]) Arrays.copyOf(elementData, size, a.getClass());
        System.arraycopy(elementData, 0, a, 0, size);
        if (a.length > size)
            a[size] = null;
        return a;
    }  

 

 

Bug详情可以参考: OpenJDK

本文参考:

https://www.cnblogs.com/zhizhizhiyuan/p/3662371.html

https://www.zhihu.com/question/66518325/answer/243761796

posted @ 2018-05-21 14:48  KristinLee  阅读(728)  评论(0编辑  收藏  举报
jQuery火箭图标返回顶部代码