java返回集合為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&lt;String&gt; 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;
}
}

 




免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM