java问题序号_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 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 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[] 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 List asList(T... a) {

return new ArrayList<>(a);

}

ArrayList(E[] array) {

a = Objects.requireNonNull(array);

}

public static 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[] 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

本文参考:

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值