• 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

  • 相关阅读:
    Gitcafe绑定自定义域名
    如何优雅地使用Sublime Text
    使用Hexo搭建专属Blog
    How to Use Android ADB Command Line Tool
    雷军北大15分钟演讲:我至少有胆量去想(转)
    浅谈android中的目录结构
    react里 MD5加密
    git忽略相应文件夹,不上传
    antd-mobile的按需加载
    当react 项目使用px2rem
  • 原文地址:https://www.cnblogs.com/Hangtutu/p/9066967.html
Copyright © 2020-2023  润新知