Why am I not getting a java.util.ConcurrentModificationException in this example?

后端 未结 10 2118
佛祖请我去吃肉
佛祖请我去吃肉 2020-11-22 07:59

Note: I am aware of the Iterator#remove() method.

In the following code sample, I don\'t understand why the List.remove in main

相关标签:
10条回答
  • 2020-11-22 08:26

    Here's why: As it is says in the Javadoc:

    The iterators returned by this class's iterator and listIterator methods are fail-fast: if the list is structurally modified at any time after the iterator is created, in any way except through the iterator's own remove or add methods, the iterator will throw a ConcurrentModificationException.

    This check is done in the next() method of the iterator (as you can see by the stacktrace). But we will reach the next() method only if hasNext() delivered true, which is what is called by the for each to check if the boundary is met. In your remove method, when hasNext() checks if it needs to return another element, it will see that it returned two elements, and now after one element was removed the list only contains two elements. So all is peachy and we are done with iterating. The check for concurrent modifications does not occur, as this is done in the next() method which is never called.

    Next we get to the second loop. After we remove the second number the hasNext method will check again if can return more values. It has returned two values already, but the list now only contains one. But the code here is:

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

    1 != 2, so we continue to the next() method, which now realizes that someone has been messing with the list and fires the exception.

    Hope that clears your question up.

    Summary

    List.remove() will not throw ConcurrentModificationException when it removes the second last element from the list.

    0 讨论(0)
  • 2020-11-22 08:28

    If you use copy-on-write collections it will work; however when you use list.iterator(), the returned Iterator will always reference the collection of elements as it was when ( as below ) list.iterator() was called, even if another thread modifies the collection. Any mutating methods called on a copy-on-write–based Iterator or ListIterator (such as add, set, or remove) will throw an UnsupportedOperationException.

    import java.util.List;
    import java.util.concurrent.CopyOnWriteArrayList;
    
    public class RemoveListElementDemo {    
        private static final List<Integer> integerList;
    
        static {
            integerList = new CopyOnWriteArrayList<>();
            integerList.add(1);
            integerList.add(2);
            integerList.add(3);
        }
    
        public static void remove(Integer remove) {
            for(Integer integer : integerList) {
                if(integer.equals(remove)) {                
                    integerList.remove(integer);
                }
            }
        }
    
        public static void main(String... args) {                
            remove(Integer.valueOf(2));
    
            Integer remove = Integer.valueOf(3);
            for(Integer integer : integerList) {
                if(integer.equals(remove)) {                
                    integerList.remove(integer);
                }
            }
        }
    }
    
    0 讨论(0)
  • 2020-11-22 08:29

    One way to handle it it to remove something from a copy of a Collection (not Collection itself), if applicable. Clone the original collection it to make a copy via a Constructor.

    This exception may be thrown by methods that have detected concurrent modification of an object when such modification is not permissible.

    For your specific case, first off, i don't think final is a way to go considering you intend to modify the list past declaration

    private static final List<Integer> integerList;
    

    Also consider modifying a copy instead of the original list.

    List<Integer> copy = new ArrayList<Integer>(integerList);
    
    for(Integer integer : integerList) {
        if(integer.equals(remove)) {                
            copy.remove(integer);
        }
    }
    
    0 讨论(0)
  • 2020-11-22 08:31

    This runs fine on Java 1.6

    ~ % javac RemoveListElementDemo.java
    ~ % java RemoveListElementDemo
    ~ % cat RemoveListElementDemo.java

    import java.util.*;
    public class RemoveListElementDemo {    
        private static final List<Integer> integerList;
    
        static {
            integerList = new ArrayList<Integer>();
            integerList.add(1);
            integerList.add(2);
            integerList.add(3);
        }
    
        public static void remove(Integer remove) {
            for(Integer integer : integerList) {
                if(integer.equals(remove)) {                
                    integerList.remove(integer);
                }
            }
        }
    
        public static void main(String... args) {                
            remove(Integer.valueOf(2));
    
            Integer remove = Integer.valueOf(3);
            for(Integer integer : integerList) {
                if(integer.equals(remove)) {                
                    integerList.remove(integer);
                }
            }
        }
    }
    

    ~ %

    0 讨论(0)
  • 2020-11-22 08:37

    This snippet will always throw a ConcurrentModificationException.

    The rule is "You may not modify (add or remove elements from the list) while iterating over it using an Iterator (which happens when you use a for-each loop)".

    JavaDocs:

    The iterators returned by this class's iterator and listIterator methods are fail-fast: if the list is structurally modified at any time after the iterator is created, in any way except through the iterator's own remove or add methods, the iterator will throw a ConcurrentModificationException.

    Hence if you want to modify the list (or any collection in general), use iterator, because then it is aware of the modifications and hence those will be handled properly.

    Hope this helps.

    0 讨论(0)
  • 2020-11-22 08:39

    In my case I did it like this:

    int cursor = 0;
    do {
        if (integer.equals(remove))
            integerList.remove(cursor);
        else cursor++;
    } while (cursor != integerList.size());
    
    0 讨论(0)
提交回复
热议问题