ConcurrentModificationException for ArrayList

133,563

Solution 1

You can't remove from list if you're browsing it with "for each" loop. You can use Iterator. Replace:

for (DrugStrength aDrugStrength : aDrugStrengthList) {
    if (!aDrugStrength.isValidDrugDescription()) {
        aDrugStrengthList.remove(aDrugStrength);
    }
}

With:

for (Iterator<DrugStrength> it = aDrugStrengthList.iterator(); it.hasNext(); ) {
    DrugStrength aDrugStrength = it.next();
    if (!aDrugStrength.isValidDrugDescription()) {
        it.remove();
    }
}

Solution 2

Like the other answers say, you can't remove an item from a collection you're iterating over. You can get around this by explicitly using an Iterator and removing the item there.

Iterator<Item> iter = list.iterator();
while(iter.hasNext()) {
  Item blah = iter.next();
  if(...) {
    iter.remove(); // Removes the 'current' item
  }
}

Solution 3

I like a reverse order for loop such as:

int size = list.size();
for (int i = size - 1; i >= 0; i--) {
    if(remove){
        list.remove(i);
    }
}

because it doesn't require learning any new data structures or classes.

Solution 4

there should has a concurrent implemention of List interface supporting such operation.

try java.util.concurrent.CopyOnWriteArrayList.class

Solution 5

While iterating through the loop, you are trying to change the List value in the remove() operation. This will result in ConcurrentModificationException.

Follow the below code, which will achieve what you want and yet will not throw any exceptions

private String toString(List aDrugStrengthList) {
        StringBuilder str = new StringBuilder();
    List removalList = new ArrayList();
    for (DrugStrength aDrugStrength : aDrugStrengthList) {
        if (!aDrugStrength.isValidDrugDescription()) {
            removalList.add(aDrugStrength);
        }
    }
    aDrugStrengthList.removeAll(removalList);
    str.append(aDrugStrengthList);
    if (str.indexOf("]") != -1) {
        str.insert(str.lastIndexOf("]"), "\n          " );
    }
    return str.toString();
}
Share:
133,563
mabuzer
Author by

mabuzer

Updated on March 08, 2020

Comments

  • mabuzer
    mabuzer about 4 years

    I have the following piece of code:

    private String toString(List<DrugStrength> aDrugStrengthList) {
        StringBuilder str = new StringBuilder();
            for (DrugStrength aDrugStrength : aDrugStrengthList) {
                if (!aDrugStrength.isValidDrugDescription()) {
                    aDrugStrengthList.remove(aDrugStrength);
                }
            }
            str.append(aDrugStrengthList);
            if (str.indexOf("]") != -1) {
                str.insert(str.lastIndexOf("]"), "\n          " );
            }
        return str.toString();
    }
    

    When I try to run it, I get ConcurrentModificationException, can anyone explain why it happens, even if the code is running in same thread? And how could I avoid it?

  • idiotgenius
    idiotgenius almost 14 years
    I had a same problem with HashMap, fixed with another implemention of Map interface. You should test it yourself.I don't know detail about CopyOnWriteArrayList
  • Hugo Gresse
    Hugo Gresse over 9 years
    the foreach syntax of java actually use Iterator, some IDE will report this solution and propose to replace with the foreach (for(MyListener listener : MyListenerList))
  • Konrad Garus
    Konrad Garus over 9 years
    @HugoGresse Yes, but this is the opposite direction. Iterator exposes remove that is safe for its iteration, something that foreach "loses".