Arrays.asList的实现

今天看Java编程思想看到容器这块的时候,发现说了Arrays.asList方法的一些特别之处,因此写篇博文来记录一下。java

Arrays是一个数组的工具类,用于操做数组。编程

public static <T> List<T> Arrays.asList(T... a) 这是asList的方法定义。意思就是传入类型为T的数组或者多个T类型的变量参数,就会返回一个List<T>的对象。数组


Java编程思想上是这么说的:"你能够使用Arrays.asList方法的输出做为List使用,可是其底层表示的是数组,所以不能调整尺寸,若是你试图用add和delete方法在列表中添加和删除元素就有可能会引起去修改数组的尺寸尝试,所以,你将会在运行时得到Unsupported Operation异常"。app

那么,首先我来来经过简单的测试来看看这句话的正确性:dom

public class ArraysTest {

    public static void main(String[] args){
        List<Integer> aList = Arrays.asList(1,2);
        for(int i : aList){
            System.out.println(i);
        }

        aList.add(3);
    }
}
看一下运行的结果:
1
2
Exception in thread "main" java.lang.UnsupportedOperationException
 at java.util.AbstractList.add(AbstractList.java:148)
 at java.util.AbstractList.add(AbstractList.java:108)
 at com.zhu.util.ArraysTest.main(ArraysTest.java:17)
 at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
 at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
 at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
 at java.lang.reflect.Method.invoke(Method.java:606)
 at com.intellij.rt.execution.application.AppMain.main(AppMain.java:134)
好吧,果不其然。异常抛出!
接下来,咱们看看源码,为何会出现这种状况
 
@SafeVarargs
public static <T> List<T> asList(T... a) {
    return new ArrayList<>(a);
}

/**
 * @serial include
 */
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;
    }
}
 Arrays.asList返回的是自己静态内部类ArrayList,而不是咱们常用的java.util.ArrayList。看看这内部类的实现,直接是将构造器传进来的数组直接做为存储数据的基础。而add和delete方法是用的是父类AbstractList的实现。

父类AbstractList中的add方法实现工具

public boolean add(E e) {
    add(size(), e);
    return true;
}
public void add(int index, E element) {
    throw new UnsupportedOperationException();
}

以此可见在AbstractList中是不支持add的实现的。 测试

相关文章
相关标签/搜索