今天學學Java中如何創(chuàng)建一個空集合以及空集合的一些使用場景和相關的坑。你可能會問呐籽,這好像沒有什么好講的狡蝶,空集合不就是new一個嘛,也就是像new ArrayList()這樣創(chuàng)建一個不久行了嗎贪惹?其實這也是一種創(chuàng)建空集合的方法,但今天小編講下通過另外一種方式創(chuàng)建空集合枫绅,以及兩種方式之間的差異。
一棵譬、通過Collections.emptyList()創(chuàng)建空集合
Java集合工具類中提供了一系列創(chuàng)建集合的靜態(tài)方法预伺,其中包括創(chuàng)建線程同步相關的Collections.synchronizedXXX()方法酬诀、空集合相關的Collections.emptyXXX()方法。通過這種方式創(chuàng)建的空集合瞒御,既然是空的,就不允許你往集合中添加元素和刪除元素趾唱,也就是不能調用相應add()和remove()方法蜻懦,我先來看看Collections類創(chuàng)建空集合的部分源代碼:
public static final List EMPTY_LIST = new EmptyList<>();......public static final List emptyList() { return (List) EMPTY_LIST;}
你會發(fā)現上面的emptyList()方法默認返回的是前面的靜態(tài)變量EMPTY_LIST,你可能會說宛乃,既然EMPTY_LIST是static的,那我直接通過Collections.EMPTY_LIST獲取不就好了征炼,沒錯谆奥,這樣做也可以眼坏,只不過在某些需要泛型的場景下宰译,調用emptyList()方法提供了相應的泛型支持擂仍。
那為什么這種方式不能添加和移除元素呢熬甚?我們來看看EmptyList內部類是怎么定義的:
// 繼承自AbstractList抽象類private static class EmptyList extends AbstractList implements RandomAccess, Serializable { private static final long serialVersionUID = 8842843931221139166L; public Iterator iterator() { return emptyIterator(); } public ListIterator 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[] 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; } @Override public boolean removeIf(Predicate filter) { Objects.requireNonNull(filter); return false; } @Override public void replaceAll(UnaryOperator operator) { Objects.requireNonNull(operator); } @Override public void sort(Comparator c) {} // Override default methods in Collection @Override public void forEach(Consumer action) { Objects.requireNonNull(action); } @Override public Spliterator spliterator() { return Spliterators.emptySpliterator(); } // Preserves singleton property private Object readResolve() { return EMPTY_LIST; }}
從上面的源代碼中我們可以發(fā)現EmptyList類并沒有重寫父類相應的add()或者remove()方法乡括,那么當調用空集合的add()方法時將默認調用AbstractList的add()方法智厌,行铣鹏,那么我們來看看父類AbstractList的add()方法是怎么實現的:
public void add(int index, E element) { throw new UnsupportedOperationException();}public E remove(int index) { throw new UnsupportedOperationException();}
很遺憾哀蘑,父類直接給你拋出UnsupportedOperationException異常,所以合溺,小編認為缀台,通過Collections創(chuàng)建的空集合不能添加或刪除元素也是合情合理的,因為是空集合嘛睛约,空哲身,那為啥還要有添加刪除操作。下面說說這種方式的使用場景膀值。
二误辑、簡單使用場景
web開發(fā)中經常使用rest + json的技術組合來進行前后端交互,那么當前端調用一個接口時翘狱,接口有可能需要返回一個空的集合給到前端砰苍,比如你根據某個條件查數據庫得不到數據時,那么此時Collections.emptyXXX()就非常合適了茬缩,因為使用new ArrayList()的初始化還會占用相關的資源吼旧。
為了說明調用add()方法會拋出異常,下面寫個小測試:
public class RemoveIfTest { private static List list = Collections.emptyList(); public static void main(String[] args) { list.add("one1"); list.add("one2"); list.add(1); list.add(2); list.add(new Object()); System.err.println(Arrays.toString(list.toArray())); }}
程序輸出:
Exception in thread "main" java.lang.UnsupportedOperationException at java.util.AbstractList.add(Unknown Source) at java.util.AbstractList.add(Unknown Source) at com.example.RemoveIfTest.main(RemoveIfTest.java:17)
三掂为、總結
總的來說,對于如何創(chuàng)建空集合的問題我們不需要糾結昼扛,重要的我們要記住通過Collections.emptyXXX()創(chuàng)建的空集合不能執(zhí)行添加刪除操作以及其中的原理欲诺,避免以后犯錯,不過其實即使你使用錯了斯稳,調試幾遍你的代碼估計也就會把問題發(fā)現出來迹恐,只不過這篇文章能幫你你省去這個發(fā)現bug的過程啦!
如果想學習Java工程化憎茂、高性能及分布式锤岸、深入淺出是偷。微服務、Spring蛋铆,MyBatis,Netty源碼分析的朋友可以加我的Java進階群:617434785留特,群里有阿里大牛直播講解技術玛瘸,以及Java大型互聯網技術的視頻免費分享給大家糊渊。