原來Java中有兩個ArrayList
首先給出一段代碼:
public class AslistMethod { public static void main(String[] args) { String sentence = "i love you"; List<String> words = Arrays.asList(sentence.split(" ")); for (int i = 0; i < words.size(); ) { if (words.get(i).equals("love")) { words.remove(i); continue; } i++; } System.out.println(words); }
然后,運行這段代碼,最后,報錯了:
Exception in thread "main" java.lang.UnsupportedOperationException at java.util.AbstractList.remove(AbstractList.java:161) at list.AslistMethod.main(AslistMethod.java:12)
錯誤鏈的棧底list.AslistMethod.main(AslistMethod.java:12)說的是這一行代碼:
words.remove(i);
這個錯誤讓我很迷茫,因為我寫這段代碼前大概看了一下,Arrays.asList方法返回的是一個ArrayList,咋就不能remove了呢?接下來,由于一個小小的失誤,把Arrays.asList的結果賦給了java.util.ArrayList,居然賦值不成功,我立馬有種瞎貓碰上死耗子的感覺。然后,就順著Arrays.asList方法一直點下去,發(fā)現(xiàn)此ArrayList非彼ArrayList。Arrays.asList返回的是java.util.Arrays.ArrayList,它只是Arrays的一個內部類。它的代碼不長,這里直接貼出來:
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) { a = Objects.requireNonNull(array); } @Override public int size() { return a.length; } @Override public Object[] toArray() { return a.clone(); } @Override @SuppressWarnings("unchecked") 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; } @Override public E get(int index) { return a[index]; } @Override public E set(int index, E element) { E oldValue = a[index]; a[index] = element; return oldValue; } @Override 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; } @Override public boolean contains(Object o) { return indexOf(o) != -1; } @Override public Spliterator<E> spliterator() { return Spliterators.spliterator(a, Spliterator.ORDERED); } }
從源碼看到,它繼承自AbstractList,并且沒有remove方法,我找到AbstractList的remove方法:
public E remove(int index) { throw new UnsupportedOperationException(); }
這就是錯誤信息的來源。最終知道,Arrays.asList返回的ArrayList并不支持remove,實際上,java.util.Arrays.ArrayList不支持任何形式的元素刪除。
總結一下,這次問題的解決確實有點兒瞎貓碰到死耗子的味道,但是,只要我認真看一下報的錯誤:
at java.util.AbstractList.remove(AbstractList.java:161)
調用java.util.AbstractList的remove方法和我預期的不一樣,它就能清晰地告訴我方向。
希望本文所述對大家學習java程序設計有所幫助。
- java正則表達式實現(xiàn)提取需要的字符并放入數(shù)組【ArrayList數(shù)組去重復功能】
- Java中ArrayList和LinkedList的遍歷與性能分析
- java Vector和ArrayList的分析及比較
- Java ArrayList 實現(xiàn)實例講解
- java ArrayList集合中的某個對象屬性進行排序的實現(xiàn)代碼
- 由ArrayList來深入理解Java中的fail-fast機制
- 分析Java中ArrayList與LinkedList列表結構的源碼
- Java中ArrayList類的用法與源碼完全解析
- Java ArrayList.toArray(T[]) 方法的參數(shù)類型是 T 而不是 E的原因分析
- java 對ArrayList進行分頁實例代碼
相關文章
Spring mvc是如何實現(xiàn)與數(shù)據(jù)庫的前后端的連接操作的?
今天給大家?guī)淼氖顷P于Spring mvc的相關知識,文章圍繞著Spring mvc是如何實現(xiàn)與數(shù)據(jù)庫的前后端的連接操作的展開,文中有非常詳細的介紹及代碼示例,需要的朋友可以參考下2021-06-06SpringBoot開發(fā)實戰(zhàn)系列之定時器
定時任務我想諸位童鞋都不陌生,簡而言之名為“設定定時鬧鐘做某件事情”,下面這篇文章主要給大家介紹了關于SpringBoot定時器的相關資料,需要的朋友可以參考下2021-08-08解讀Spring接口方法加@Transactional失效的原因
這篇文章主要介紹了Spring接口方法加@Transactional失效的原因解讀,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-03-03