一个Arrays.asList()基本的用法:
String[] str = new String[]{"1","2","3"}; ArrayList al = new ArrayList(Arrays.asList(str));//将数组元素添加到集合的一种快捷方式
有时候这样使用就有可能出现问题:
String[] str = new String[]{"1","2","3"}; List aslist = Arrays.asList(str);
aslist是List类型的对象,但是调用List接口的方法的时候会出错就像这样:
aslsit.add("4"); Exception in thread "main" java.lang.UnsupportedOperationException at java.util.AbstractList.add(Unknown Source) at java.util.AbstractList.add(Unknown Source) at test.LinkedListTest.main(LinkedListTest.java:13)
出现这个错误的原因是没有定义add()方法的具体实现,这些异常都在AbstractList抛出,看一下Arrays源码:
@SafeVarargs public static <T> List<T> asList(T... a) { return new ArrayList<>(a); }
没有问题啊,返回的是一个ArrayList,但是这个ArrayList不是ArrayList.class而是Arrays$ArrayList这样一个内部类
private static class ArrayList<E> extends AbstractList<E> implements RandomAccess, java.io.Serializable { private static final long serialVersionUID = -2764017481108945198L; private final E[] a; ArrayList(E[] array) { if (array==null) throw new NullPointerException(); a = array; } public int size() { return a.length; } public Object[] toArray() { return a.clone(); } public <T> T[] toArray(T[] a) { int size = size(); if (a.length < size) return Arrays.copyOf(this.a, size, (Class<? extends T[]>) a.getClass()); System.arraycopy(this.a, 0, a, 0, size); if (a.length > size) a[size] = null; return a; } public E get(int index) { return a[index]; } public E set(int index, E element) { E oldValue = a[index]; a[index] = element; return oldValue; } public int indexOf(Object o) { if (o==null) { for (int i=0; i<a.length; i++) if (a[i]==null) return i; } else { for (int i=0; i<a.length; i++) if (o.equals(a[i])) return i; } return -1; } public boolean contains(Object o) { return indexOf(o) != -1; } }
这个内部类中并没有实现list的一些方法,所以调用add(),remove()这些方法都会出现错误。
这个内部类中有一个泛型的数组private final E[] a,所以Arrays.asList返回的集合中的数组其实是有具体的类型的,而不是Object[]。
相当于List<String> list = new Arrays$ArrayList<String>();
但是有类型的数组在转换过程中就会出现某些问题,有个Bug就是这样引起的。具体见 《c.toArray might not return Object[]》。
时间: 2024-11-08 07:51:57