集合 iterator.remove()方法詳解

直接上代碼:

public class test {
    public static void main(String[] args) {
        List<Integer> list = new ArrayList<>();
        for (int i = 0 ; i < 10 ; i++ ) {
            list.add(i);
        }
        Iterator<Integer> iterator = list.iterator();
        int i = 0 ;
        while(iterator.hasNext()) {

            if (i == 3) {
                iterator.remove(); //報java.lang.IllegalStateException異常
            }
            i ++;
        }
        System.out.println(list);
    }
}

爲什麼會報異常呢,通過一些查資料或者基礎較好的讀者知道只需要使用調用迭代器iterator.next()方法即可返回當前元素,所以只需要在remove()方法前面加上

iterator.next();

注意這裏有人會使用int x = iterator.next();這樣來把當前的索引賦值然後再進行操作,其實這樣是沒必要的,我們來根據這個例子深入的瞭解一下iterator的源碼是如何實現的。

首先我們調用的是list.iterator()方法來獲取這個迭代器,當然list類並沒有實現這個方法,我們最終是使用它的子類ArrayList來獲取迭代器:

/**
     * Returns an iterator over the elements in this list in proper sequence.
     *
     * <p>The returned iterator is <a href="#fail-fast"><i>fail-fast</i></a>.
     *
     * @return an iterator over the elements in this list in proper sequence
     */
    public Iterator<E> iterator() {
        return new Itr();
    }

    /**
     * An optimized version of AbstractList.Itr
     */
    private class Itr implements Iterator<E> {
        int cursor;       // index of next element to return
        int lastRet = -1; // index of last element returned; -1 if no such
        int expectedModCount = modCount;

        Itr() {}

        public boolean hasNext() {
            return cursor != size;
        }

        @SuppressWarnings("unchecked")
        public E next() {
            checkForComodification();
            int i = cursor;
            if (i >= size)
                throw new NoSuchElementException();
            Object[] elementData = ArrayList.this.elementData;
            if (i >= elementData.length)
                throw new ConcurrentModificationException();
            cursor = i + 1;
            return (E) elementData[lastRet = i];
        }

        public void remove() {
            if (lastRet < 0)
                throw new IllegalStateException();
            checkForComodification();

            try {
                ArrayList.this.remove(lastRet);
                cursor = lastRet;
                lastRet = -1;
                expectedModCount = modCount;
            } catch (IndexOutOfBoundsException ex) {
                throw new ConcurrentModificationException();
            }
        }

可以看到,iterator()方法返回了一個Itr對象,接着下面,Itr是一個內部類,並且實現了Itertor接口。

回到我們最初的問題,爲什麼在迭代器中不用在不使用next()方法情況下進行remove操作會出錯?我們來看下remove()方法源碼,

在remove方法第一行,即可看到:

  if (lastRet < 0)
                throw new IllegalStateException();

當lastRet < 0的情況下,會拋出IllegalStateException異常,那麼這個lastRet是什麼呢,其實在Itr類就定義了這個lastRet變量,和它一起的還有cursor,expectedModCount;

  private class Itr implements Iterator<E> {
        int cursor;       // index of next element to return
        int lastRet = -1; // index of last element returned; -1 if no such
        int expectedModCount = modCount;

 通過註釋可以看到,lastRet就是上個元素的索引,默認是-1,所以直接調用迭代器的remove()方法會報錯就是這個原因,

所以在上面的next()方法裏,我們可以看到這樣的代碼:

            int i = cursor;
            if (i >= size)
                throw new NoSuchElementException();
            Object[] elementData = ArrayList.this.elementData;
            if (i >= elementData.length)
                throw new ConcurrentModificationException();
            cursor = i + 1;
            return (E) elementData[lastRet = i];

它會把cursor遊標賦值給i,然後進行+1操作,最後將lastRet賦值成i,所以lastRet就不是默認的-1了,而是會得到上個元素的索引,

所以綜合上述的結論:

在使用迭代器iterator.remove()方法前,一定先調用next()方法來給變量lastRet進行賦值操作;

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章