I have a bunch of Runnable objects that are launched from a base class. These objects are iterating through and deleting items in a shared ArrayList at random times. I have synchronized both methods but am getting a ConcurrentModicationException
. I believe its because they are synchronized but are not synchronized on each other. Is this the case? If so, which class do I acquire the lock on?
Environment Class:
class Environment{
ArrayList<Critter> critter_array = new ArrayList<Critter>();
ArrayList<Food> food_array = new ArrayList<Food>();
Environment(){
Executor ex = Executors.newCachedThreadPool();
Critter A = new Critter();
Critter B = new Critter();
critter_array.add(A);
critter_array.add(B);
ex.execute(A);
ex.execute(B);
}
public synchronized void destroyFood(Food item){
Iterator<Food> iter = food_array.iterator();
while(iter.hasNext()){
Food temp = iter.next();
food_array.remove(temp);
break;
}
}
}
Critter class:
class Critter implements Runnable{
Environment envi;
Critter(Environment E){
envi = E;
}
@Override
public void run() {
//do other stuff
CritterUtilities.iteratorMethod(this, envi);
}
}
CritterUtilities class:
class CritterUtilities{
public static synchronized iteratorMethod(Critter self, Environment envi){
Iterator<OtherObject> iter = envi.getFood().listIterator();
while(iter.hasNext()){
Food temp = iter.next(); /////////Problem is here
//other stuff that's not related to the prob
}
}
}
Stack Trace:
Exception in thread "pool-1-thread-2" java.util.ConcurrentModificationException
at java.util.AbstractList$Itr.checkForComodification(Unknown Source)
at java.util.AbstractList$Itr.next(Unknown Source)
at CritterUtil.getViewableFood(CritterUtil.java:28)
at CritterUtil.getNearestFood(CritterUtil.java:41)
at Critter.hunt(Critter.java:163)
at Critter.run(Critter.java:139)
at java.util.concurrent.ThreadPoolExecutor$Worker.runTask(Unknown Source)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(Unknown Source)
at java.lang.Thread.run(Unknown Source)