Collections.emptyList()引发的java.lang.UnsupportedOperationException

探讨了使用Collections.emptyList()创建空列表时,addAll方法的行为及其实现原理,并提供了一个避免异常的示例。

缘由:

private static List<Integer> getEmptyList() {
return Collections.emptyList();
}

public static void main(String[] args) {
try {
List<Integer> list1 = new ArrayList<>(Arrays.asList(1,2,3,4,5));
List<Integer> list2 = getEmptyList();
list2.addAll(list1);
System.out.println(list1.toString());
} catch (Exception e) {
e.printStackTrace();
}
}


此方法会抛出以下异常:

java.lang.UnsupportedOperationException
at java.util.AbstractList.add(Unknown Source)
at java.util.AbstractList.add(Unknown Source)
at java.util.AbstractCollection.addAll(Unknown Source)

...


原因在于


ArrayList里面有addAll的实现, 详见源码

    public boolean addAll(Collection<? extends E> c) {
        Object[] a = c.toArray();
        int numNew = a.length;
        ensureCapacityInternal(size + numNew);  // Increments modCount
        System.arraycopy(a, 0, elementData, size, numNew);
        size += numNew;
        return numNew != 0;
    }

而emptyList里面没有, 详见源码

/**
     * @serial include
     */
    private static class EmptyList<E>
        extends AbstractList<E>
        implements RandomAccess, Serializable {
        private static final long serialVersionUID = 8842843931221139166L;


        public Iterator<E> iterator() {
            return emptyIterator();
        }
        public ListIterator<E> listIterator() {
            return emptyListIterator();
        }


        public int size() {return 0;}
        public boolean isEmpty() {return true;}


        public boolean contains(Object obj) {return false;}
        public boolean containsAll(Collection<?> c) { return c.isEmpty(); }


        public Object[] toArray() { return new Object[0]; }


        public <T> T[] toArray(T[] a) {
            if (a.length > 0)
                a[0] = null;
            return a;
        }


        public E get(int index) {
            throw new IndexOutOfBoundsException("Index: "+index);
        }


        public boolean equals(Object o) {
            return (o instanceof List) && ((List<?>)o).isEmpty();
        }


        public int hashCode() { return 1; }


        // Preserves singleton property
        private Object readResolve() {
            return EMPTY_LIST;
        }
    }


但是, 如果我们这么写就可以避免这个异常

private static List<Integer> getEmptyList() {
return Collections.emptyList();
}

public static void main(String[] args) {
try {
List<Integer> list1 = new ArrayList<>(Arrays.asList(1,2,3,4,5));
List<Integer> list2 = getEmptyList();
list1.addAll(list2);
System.out.println(list1.toString());
} catch (Exception e) {
e.printStackTrace();
}
}


程序可以输出

[1, 2, 3, 4, 5]



所以以后用 Collections.emptyList() 的时候要注意addAll的操作

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值