鍍金池/ 教程/ Java/ Java 集合細節(jié)(二):asList 的缺陷
Java 集合細節(jié)(四):保持 compareTo 和 equals 同步
Iterator
使用序列化實現(xiàn)對象的拷貝
fail-fast 機制
關(guān)鍵字 final
Vector
HashTable
Java 集合細節(jié)(一):請為集合指定初始容量
強制類型轉(zhuǎn)換
數(shù)組之一:認(rèn)識 JAVA 數(shù)組
Java 集合細節(jié)(三):subList 的缺陷
hashCode
ArrayList
數(shù)組之二
List 總結(jié)
LinkedList
Java 提高篇(九)—–實現(xiàn)多重繼承
Java 的四舍五入
關(guān)鍵字 static
理解 Java 的三大特性之多態(tài)
抽象類與接口
集合大家族
異常(二)
Java 集合細節(jié)(二):asList 的缺陷
Map 總結(jié)
TreeSet
equals() 方法總結(jié)
Java 提高篇(十)—–詳解匿名內(nèi)部類
HashMap
Stack
詳解內(nèi)部類
TreeMap
異常(一)
詳解 Java 定時任務(wù)
HashSet
字符串
理解 Java 的三大特性之繼承
理解 Java 的三大特性之封裝
代碼塊

Java 集合細節(jié)(二):asList 的缺陷

在實際開發(fā)過程中我們經(jīng)常使用 asList 講數(shù)組轉(zhuǎn)換為 List,這個方法使用起來非常方便,但是 asList 方法存在幾個缺陷:

一、避免使用基本數(shù)據(jù)類型數(shù)組轉(zhuǎn)換為列表

使用 8 個基本類型數(shù)組轉(zhuǎn)換為列表時會存在一個比較有味的缺陷。先看如下程序:


    public static void main(String[] args) {
            int[] ints = {1,2,3,4,5};
            List list = Arrays.asList(ints);
            System.out.println("list'size:" + list.size());
        }
        ------------------------------------
        outPut:
        list'size:1

程序的運行結(jié)果并沒有像我們預(yù)期的那樣是 5 而是逆天的 1,這是什么情況?先看源碼:


    public static <T> List<T> asList(T... a) {
            return new ArrayList<>(a);
        }

asList 接受的參數(shù)是一個泛型的變長參數(shù),我們知道基本數(shù)據(jù)類型是無法發(fā)型化的,也就是說 8 個基本類型是無法作為 asList 的參數(shù)的, 要想作為泛型參數(shù)就必須使用其所對應(yīng)的包裝類型。但是這個這個實例中為什么沒有出錯呢?因為該實例是將 int 類型的數(shù)組當(dāng)做其參數(shù),而在 Java 中數(shù)組是一個對象,它是可以泛型化的。所以該例子是不會產(chǎn)生錯誤的。既然例子是將整個 int 類型的數(shù)組當(dāng)做泛型參數(shù),那么經(jīng)過 asList 轉(zhuǎn)換就只有一個 int 的列表了。如下:


    public static void main(String[] args) {
        int[] ints = {1,2,3,4,5};
        List list = Arrays.asList(ints);
        System.out.println("list 的類型:" + list.get(0).getClass());
        System.out.println("list.get(0) == ints:" + list.get(0).equals(ints));
    }
    --------------------------------------------
    outPut:
    list 的類型:class [I
    list.get(0) == ints:true

從這個運行結(jié)果我們可以充分證明 list 里面的元素就是 int 數(shù)組。弄清楚這點了,那么修改方法也就一目了然了:將 int 改變?yōu)?Integer。


    public static void main(String[] args) {
            Integer[] ints = {1,2,3,4,5};
            List list = Arrays.asList(ints);
            System.out.println("list'size:" + list.size());
            System.out.println("list.get(0) 的類型:" +  list.get(0).getClass());
            System.out.println("list.get(0) == ints[0]:" + list.get(0).equals(ints[0]));
        }
        ----------------------------------------
        outPut:
        list'size:5
        list.get(0) 的類型:class java.lang.Integer
        list.get(0) == ints[0]:true

Java 細節(jié)(2.1):在使用 asList 時不要將基本數(shù)據(jù)類型當(dāng)做參數(shù)。

二、asList 產(chǎn)生的列表不可操作

對于上面的實例我們再做一個小小的修改:


    public static void main(String[] args) {
            Integer[] ints = {1,2,3,4,5};
            List list = Arrays.asList(ints);
            list.add(6);
        }

該實例就是講 ints 通過 asList 轉(zhuǎn)換為 list 類別,然后再通過 add 方法加一個元素,這個實例簡單的不能再簡單了,但是運行結(jié)果呢?打出我們所料:


    Exception in thread "main" java.lang.UnsupportedOperationException
        at java.util.AbstractList.add(Unknown Source)
        at java.util.AbstractList.add(Unknown Source)
        at com.chenssy.test.arrayList.AsListTest.main(AsListTest.java:10)

運行結(jié)果盡然拋出 UnsupportedOperationException 異常,該異常表示 list 不支持 add 方法。這就讓我們郁悶了,list 怎么可能不支持 add 方法呢?難道 JDK 腦袋堵塞了?我們再看 asList 的源碼:


    public static <T> List<T> asList(T... a) {
            return new ArrayList<>(a);
        }

asList 接受參數(shù)后,直接 new 一個 ArrayList,到這里看應(yīng)該是沒有錯誤的???別急,再往下看:


    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;
            }
            //.................
        }

這是 ArrayList 的源碼,從這里我們可以看出,此 ArrayList 不是 java.util.ArrayList,他是 Arrays 的內(nèi)部類。該內(nèi)部類提供了 size、toArray、get、set、indexOf、contains 方法,而像 add、remove 等改變 list 結(jié)果的方法從 AbstractList 父類繼承過來,同時這些方法也比較奇葩,它直接拋出 UnsupportedOperationException 異常:


    public boolean add(E e) {
            add(size(), e);
            return true;
        }

        public E set(int index, E element) {
            throw new UnsupportedOperationException();
        }

        public void add(int index, E element) {
            throw new UnsupportedOperationException();
        }

        public E remove(int index) {
            throw new UnsupportedOperationException();
        }

通過這些代碼可以看出 asList 返回的列表只不過是一個披著 list 的外衣,它并沒有 list 的基本特性(變長)。該 list 是一個長度不可變的列表,傳入?yún)?shù)的數(shù)組有多長,其返回的列表就只能是多長。所以:

Java 細節(jié)(2.2):不要試圖改變 asList 返回的列表,否則你會自食苦果。

上一篇:HashTable下一篇:字符串