1. 程式人生 > >java集合遍歷刪除指定元素異常分析總結

java集合遍歷刪除指定元素異常分析總結

它的 一次 但是 代碼 元素 拋出異常 源碼 刪除指定元素 test

在使用集合的過程中,我們經常會有遍歷集合元素,刪除指定的元素的需求,而對於這種需求我們往往使用會犯些小錯誤,導致程序拋異常或者與預期結果不對,本人很早之前就遇到過這個坑,當時沒註意總結,結果前段時間又遇到了這個問題,因此,總結下遍歷集合的同時如何刪除集合中指定的元素;

1.錯誤場景復原

public class ListRemoveTest {

	public static void main(String[] args) {
		List<User> users = new ArrayList<User>();
		users.add(new User("liu1",24));
		users.add(new User("liu2",24));
		users.add(new User("liu3",24));
		users.add(new User("liu4",24));
		
		Iterator<User> iterator = users.iterator();
		while(iterator.hasNext()) {
			User user = iterator.next();
			if(user.getName().equals("liu2")) {
				users.remove(user);
} System.out.println(user); } } }

或者如下代碼

public class ListRemoveTest {

	public static void main(String[] args) {
		List<User> users = new ArrayList<User>();
		users.add(new User("liu1",24));
		users.add(new User("liu2",24));
		users.add(new User("liu3",24));
		users.add(new User("liu4",24));
		
	        for (User user : users) {
			if(user.getName().equals("liu2")) {
				users.remove(user);
			}
			System.out.println(user);
		}
	}
}

以上兩種用法都會跑出如下異常:

技術分享圖片

2.原因分析

上面兩種錯誤,我想很多人都遇到過,這是我們很容易犯的錯誤,但是為啥會出現上述異常呢,我們又該如何正確遍歷集合的同時,刪除指定的元素呢!

2.1 原因解析

首先,對於foreach循環遍歷,本質上還是叠代器的模式,上面的for語句等價於如下代碼:

for (Iterator<User> iterator = users.iterator(); iterator.hasNext();) {
	User user = iterator.next();
	if(user.getName().equals("liu2")) {
		users.remove(user);
	}
	System.out.println(user);
}

因此,上述錯誤的本質,就要看叠代器iterator的源碼啦

在ArrayList中,它的修改操作(add/remove)都會對modCount這個字段+1,modCount可以看作一個版本號,每次集合中的元素被修改後,都會+1(即使溢出)。

  public boolean remove(Object o) {
        if (o == null) {
            for (int index = 0; index < size; index++)
                if (elementData[index] == null) {
                    fastRemove(index);
                    return true;
                }
        } else {
            for (int index = 0; index < size; index++)
                if (o.equals(elementData[index])) {
                    fastRemove(index);
                    return true;
                }
        }
        return false;
    }

 private void fastRemove(int index) {
        modCount++;
        int numMoved = size - index - 1;
        if (numMoved > 0)
            System.arraycopy(elementData, index+1, elementData, index,
                             numMoved);
        elementData[--size] = null; // clear to let GC do its work
    }

接下來再看看AbsrtactList中iteraor方法

public Iterator<E> iterator() {
    return new Itr();
}

它返回一個內部類,這個類實現了iterator接口,代碼如下:

private class Itr implements Iterator<E> {
    int cursor = 0;

    int lastRet = -1;

    int expectedModCount = modCount;

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

    public E next() {
        checkForComodification();
        try {
            E next = get(cursor);
            lastRet = cursor++;
            return next;
        } catch (IndexOutOfBoundsException e) {
            checkForComodification();
            throw new NoSuchElementException();
        }
    }

    public void remove() {
        if (lastRet == -1)
            throw new IllegalStateException();
        checkForComodification();

        try {
            AbstractList.this.remove(lastRet);
            if (lastRet < cursor)
                cursor--;
            lastRet = -1;
            // 修改expectedModCount 的值
            expectedModCount = modCount;
            } catch (IndexOutOfBoundsException e) {
            throw new ConcurrentModificationException();
        }
    }

    final void checkForComodification() {
        if (modCount != expectedModCount)
            throw new ConcurrentModificationException();
    }
    }

在內部類Itr中,有一個字段expectedModCount ,初始化時等於modCount,即當我們調用list.iterator()返回叠代器時,該字段被初始化為等於modCount。在類Itr中next/remove方法都有調用checkForComodification()方法,在該方法中檢測modCount == expectedModCount,如果不相等則拋出ConcurrentModificationException。

前面說過,在集合的修改操作(add/remove)中,都對modCount進行了+1。

在叠代過程中,執行list.remove(val),使得modCount+1,當下一次循環時,執行 it.next(),checkForComodification方法發現modCount != expectedModCount,則拋出異常。

2.2 預期結果不對,但是不拋異常

註意:還有一種更坑的場景,當刪除集合的倒數第二個元素時,程序不會拋出任何異常,只是結果與預期的不相符,如果在應用過程中不認真觀察,很難發現該錯誤!

錯誤實例如下:

public static void main(String[] args) {
	List<User> users = new ArrayList<User>();
	users.add(new User("liu1",24));
users.add(new User("liu2",24)); users.add(new User("liu3",24));
users.add(new User("liu4",24)); Iterator<User> iterator = users.iterator();
while(iterator.hasNext()) {
User user = iterator.next();
if(user.getName().equals("liu3")) {
users.remove(user);
}
System.out.println(user);
} }

運行結果如下:

技術分享圖片

遍歷過程刪除了倒數第二個元素,那麽最後一個元素就永遠遍歷不到了,這個主要原因就是Iterator源碼中hasNext方法中,判斷當前元素下標和集合大小是否相等

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

當刪除倒數第二個元素後,當前元素下標和集合的大小相等了,跳出了循環,就會遍歷最後一個集合元素了;

3.正確用法

要想在集合遍歷的過程中刪除指定元素,就務必使用叠代器自身的remove方法;
再來看看內部類Itr的remove()方法,在刪除元素後,有這麽一句expectedModCount = modCount,同步修改expectedModCount 的值。所以,如果需要在使用叠代器叠代時,刪除元素,可以使用叠代器提供的remove方法。 其他集合(Map/Set)使用叠代器叠代也是一樣。

所以 Iterator 在工作的時候是不允許被叠代的對象被改變的。
但你可以使用 Iterator 本身的方法 remove() 來刪除對象, Iterator.remove() 方法會在刪除當前叠代對象的同時維護索引的一致

具體正確用法代碼如下:

public class ListRemoveTest {

public static void main(String[] args) {
List<User> users = new ArrayList<User>();
users.add(new User("liu1",24));
users.add(new User("liu2",24));
users.add(new User("liu3",24));
users.add(new User("liu4",24));

Iterator<User> iterator = users.iterator();
while(iterator.hasNext()) {
User user = iterator.next();
if(user.getName().equals("liu2")) {
iterator.remove();
}
System.out.println(user);
}
System.out.println(users);
}
}

運行結果如下:

技術分享圖片

與預期結果一致;

java集合遍歷刪除指定元素異常分析總結