java返回集合為null還是空集合及空集合的三種寫法小結
返回集合為null還是空集合及空集合的三種寫法
個人認為在自己寫接口時,需要返回集合時返回一個空集合,比如mybatis查詢如果返回一個集合,結果為空時也會返回一個空集合而不是null。
那麼這樣有什麼好處呢?最大的好處就是調用方不用在判斷是否為null,可以直接用,因為不用拋空指針。
當然這也有缺點,如果返回Lists.newArrayList();或者new ArrayList();這會新建一個對象,而這個對象很可能是沒必要的,這樣白白浪費性能。
解決方法當然也有,可以用Collections.emptyList();這個方法返回一個空集合,並不會新建對象,而是返回
public static final List EMPTY_LIST = new EmptyList<>();
這個變量。
當然這也有缺點,如果調用方隻是遍歷,這沒什麼不會報錯,但是如果要新增,刪除裡面的元素那就會報錯,
那麼你可能想為什麼,原因就是代碼裡直接寫死瞭調用時報錯,那麼為什麼要這樣寫呢?
原因也很簡單,如果多個線程對這個集合增刪,那麼調用方就全亂瞭,所以采用瞭直接報錯,快速失敗的方法
來解決問題。
總結:
返回null,返回new ArrayList<>(),返回EMPTY_LIST 。
null肯定是不推薦的,那麼是新建一個List還是返回空List呢?
這要根據接口的性能要求,如果性能要求高返回EMPTY_LIST,否則新建一個對象。
返回空List的方式
方式一:new ArrayList()
JDK1.8已經優化瞭,默認構造函數創建的list內部共享空數組,首次插入數據時才會擴容到默認容量;
private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {}; public ArrayList() { this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA; }
方式二:new ArrayList(0)
private static final Object[] EMPTY_ELEMENTDATA = {}; public ArrayList(int initialCapacity) { if (initialCapacity > 0) { this.elementData = new Object[initialCapacity]; } else if (initialCapacity == 0) { this.elementData = EMPTY_ELEMENTDATA; } else { throw new IllegalArgumentException("Illegal Capacity: "+ initialCapacity); } }
方式三:Collections.emptyList()(推薦)
特點:不可變,安全
/** * The empty list (immutable). This list is serializable. * * @see #emptyList() */ @SuppressWarnings("rawtypes") public static final List EMPTY_LIST = new EmptyList<>(); /** * Returns an empty list (immutable). This list is serializable. * * <p>This example illustrates the type-safe way to obtain an empty list: * <pre> * List<String> s = Collections.emptyList(); * </pre> * * @implNote * Implementations of this method need not create a separate <tt>List</tt> * object for each call. Using this method is likely to have comparable * cost to using the like-named field. (Unlike this method, the field does * not provide type safety.) * * @param <T> type of elements, if there were any, in the list * @return an empty immutable list * * @see #EMPTY_LIST * @since 1.5 */ @SuppressWarnings("unchecked") public static final <T> List<T> emptyList() { return (List<T>) EMPTY_LIST; } /** * @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; } @Override public boolean removeIf(Predicate<? super E> filter) { Objects.requireNonNull(filter); return false; } @Override public void replaceAll(UnaryOperator<E> operator) { Objects.requireNonNull(operator); } @Override public void sort(Comparator<? super E> c) { } // Override default methods in Collection @Override public void forEach(Consumer<? super E> action) { Objects.requireNonNull(action); } @Override public Spliterator<E> spliterator() { return Spliterators.emptySpliterator(); } // Preserves singleton property private Object readResolve() { return EMPTY_LIST; } }
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。
推薦閱讀:
- Java Collections的emptyList、EMPTY_LIST詳解與使用說明
- Java中Collections.emptyList()的註意事項
- JDK1.8中ArrayList是如何擴容的
- 關於ArrayList的動態擴容機制解讀
- Java Collections.EMPTY_LIST與Collections.emptyList()的區別