Trigger case

Requirement: Perform a traversal of the collection and delete the a element in the collection. Many people would write the following code, but run it to throw an exception.

List<String> list = new ArrayList<>();
list.add("a");
list.add("b");
list.add("c");

for (String str : list) {
    if ("a".equals(str)) {
        list.remove("a"); }}Copy the code

Can appear unusually ConcurrentModificationException concurrent modification, so why this anomaly?

Error reason

The above code ConcurrentModificationException real causes is modCount inconsistencies. Iterator’s next() method tracks modCount as it iterates through the ArrayList. If by adding or removing elements to modify the collection, modCount will change, and with the expected modCount mismatch, therefore, the Iterator will throw ConcurrentModificationException

The source code parsing

ArrayList has a modCount member variable, derived from AbstractList

protected transient int modCount = 0;
Copy the code

ModCount records the number of structural changes to the collection.

Structural modification is an operation that adds, removes one or more elements, or visibly resizes the array behind them. Simply changing the content of an element is not called structural change.

/ / add
public boolean add(E e) {
    ensureCapacityInternal(size + 1);  // Call the method inside modCount+1
    elementData[size++] = e;
    return true;
}

// Add at the specified location
public void add(int index, E element) {
    rangeCheckForAdd(index);
    ensureCapacityInternal(size + 1);  // Call the method inside modCount+1
    System.arraycopy(elementData, index, elementData, index + 1,
                     size - index);
    elementData[index] = element;
    size++;
}

private void ensureCapacityInternal(int minCapacity) {
    ensureExplicitCapacity(calculateCapacity(elementData, minCapacity));
}

private void ensureExplicitCapacity(int minCapacity) {
    modCount++;   // modCount changes

    if (minCapacity - elementData.length > 0)
        grow(minCapacity);
}

// Drop the object at the specified index position
public E remove(int index) {
    rangeCheck(index);  // Check if the index exists

    modCount++; 		// modCount+1
    E oldValue = elementData(index);
    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

    return oldValue;
}
/ / delete
public boolean remove(Object o) {
    if (o == null) {
        for (int index = 0; index < size; index++)
            if (elementData[index] == null) {
                fastRemove(index);    // modCount+1 inside the called method
                return true; }}else {
        for (int index = 0; index < size; index++)
            if (o.equals(elementData[index])) {
                fastRemove(index);	// modCount+1 inside the called method
                return true; }}return false;
}

 private void fastRemove(int index) {
     modCount++;			// modCount changes
     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
 }
/ /...
Copy the code

As you can see from the ArrayList source code, modCount changes when collections are added or deleted (moCount++). Other scenarios where modCount changes can be viewed with the ArrayList method.

Why so modCount change can cause abnormal ConcurrentModificationException concurrent modification?

When a program performs a forEach walk (as in this case), it actually calls Itertor’s hasNext method and then next.

Itertor maintains cursor, lastRet, expectedModCount three member variables.

private class Itr implements Iterator<E> {
    int cursor;       // The index of the element to be fetched
    int lastRet = -1; // The index of the last fetched element
    int expectedModCount = modCount;  // Records the number of times the collection structure changed when the iterator was created
	/ /...
}
Copy the code

HasNext () method: determines whether traversal can continue

If cursor=size, subsequent values or deletes will cross the boundary
public boolean hasNext(a) {
    returncursor ! = size; }Copy the code

Next method: Fetch the data

public E next(a) {
    / * * * determine whether the expectedModCount with modCount between equal, if not equal, the thrown * concurrentModificationException * * /
    checkForComodification();
    int i = cursor;   // Assign the index of the element to be fetched,
    if (i >= size)
        throw new NoSuchElementException();
    Object[] elementData = ArrayList.this.elementData;
    if (i >= elementData.length)
        throw new ConcurrentModificationException();
    cursor = i + 1;    // Add one to make the cursor become the index of the value to be taken in the next iteration
    return (E) elementData[lastRet = i]; // Assign a value to lastRet, at which point I becomes the index of the last fetched value
}

final void checkForComodification(a) {
    if(modCount ! = expectedModCount)// Run an exception when it is not equal
        throw new ConcurrentModificationException();
}
Copy the code

When the remove method is executed during traversal, the modCount changes modCount++, but the expectedModCount does not change. So when performing the next method checkForComodification, modCount++ and expectedModCount are not equal, will throw ConcurrentModificationException.

The solution

1. Use iterators directly

Iterator<String> iterator = list.iterator();
while (iterator.hasNext()) {
    if ("a".equals(iterator.next())) { iterator.remove(); }}Copy the code

Use the remove of the Iterator method here, so why is it delete won’t produce ConcurrentModificationException?

public void remove(a) {
    // lastRet defaults to -1. To use remove, you must first assign lastRet to the next method
    if (lastRet < 0)
        throw new IllegalStateException();
    ExpectedModCount = expectedModCount = expectedModCount = expectedModCount
    checkForComodification();
    try {
        ArrayList.this.remove(lastRet);    // Delete the element fetched from the last next
        cursor = lastRet;	// Fix to assign the index of the deleted element to the index of the next fetched element, otherwise one bit will be missed
        lastRet = -1;		// The index of the last fetched element is reset to -1, and the value will be assigned next time to prevent deduplication
        expectedModCount = modCount;  // Assign modCount to the expectedModCount, keep it equal
    } catch (IndexOutOfBoundsException ex) {
        throw newConcurrentModificationException(); }}Copy the code

2, use,Streamfiltration

List<String> collect = list
    .stream()
    .filter(str -> !"a".equals(str))
    .collect(Collectors.toList());
Copy the code

3, use,removeIf()

list.removeIf(str -> "a".equals(str));
Copy the code

4, etc.

conclusion

  • ConcurrentModfiicationExceptionThe reason is thatmodCountDon’t agree
  • Cannot call the collection directly while traversing the collectionremoveMethod, which causes a structural change in the set
  • ArrayListIt is thread unsafe when a thread goes to changemodCountOther threads also have errors