当前位置:  开发笔记 > 编程语言 > 正文

迭代集合,在循环中删除对象时避免使用ConcurrentModificationException

如何解决《迭代集合,在循环中删除对象时避免使用ConcurrentModificationException》经验,为你挑选了11个好方法。

我们都知道你不能这样做:

for (Object i : l) {
    if (condition(i)) {
        l.remove(i);
    }
}

ConcurrentModificationException等等......这显然有时起作用,但并非总是如此.这是一些特定的代码:

public static void main(String[] args) {
    Collection l = new ArrayList<>();

    for (int i = 0; i < 10; ++i) {
        l.add(4);
        l.add(5);
        l.add(6);
    }

    for (int i : l) {
        if (i == 5) {
            l.remove(i);
        }
    }

    System.out.println(l);
}

当然,这会导致:

Exception in thread "main" java.util.ConcurrentModificationException

...即使多线程没有这样做......无论如何.

什么是这个问题的最佳解决方案?如何在循环中从集合中删除项而不抛出此异常?

我也在Collection这里使用任意,不一定是ArrayList,所以你不能依赖get.



1> Bill K..:

Iterator.remove() 是安全的,你可以像这样使用它:

List list = new ArrayList<>();

// This is a clever way to create the iterator and call iterator.hasNext() like
// you would do in a while-loop. It would be the same as doing:
//     Iterator iterator = list.iterator();
//     while (iterator.hasNext()) {
for (Iterator iterator = list.iterator(); iterator.hasNext();) {
    String string = iterator.next();
    if (string.isEmpty()) {
        // Remove the current element from the iterator and the list.
        iterator.remove();
    }
}

请注意,这Iterator.remove()是在迭代期间修改集合的唯一安全方法; 如果在迭代进行过程中以任何其他方式修改基础集合,则行为未指定.

来源:docs.oracle>集合界面


同样,如果你有一个ListIterator并且想要添加项目,你可以使用ListIterator#add,出于同样的原因你可以使用Iterator#remove 它 - 它的设计允许它.


你的情况,你想从列表中删除,但同样的限制,如果想put成为一个Map在迭代的内容.


如果要删除当前迭代中返回的元素以外的元素,该怎么办?

2> Claudiu..:

这有效:

Iterator iter = l.iterator();
while (iter.hasNext()) {
    if (iter.next() == 5) {
        iter.remove();
    }
}

我假设因为foreach循环是用于迭代的语法糖,使用迭代器无济于事......但它给你这个.remove()功能.


foreach循环*是*用于迭代的语法糖.但是正如您所指出的,您需要在迭代器上调用remove - foreach不允许您访问.因此你无法在foreach循环中删除的原因(即使你*实际上是在引擎盖下使用迭代器)
+1例如代码在上下文中使用iter.remove(),而Bill K的回答并没有[直接].

3> assylias..:

使用Java 8,您可以使用新removeIf方法.适用于您的示例:

Collection coll = new ArrayList<>();
//populate

coll.removeIf(i -> i == 5);


OOOOO!我希望Java 8或9中的某些内容可能有所帮助.这对我来说似乎相当冗长,但我仍然喜欢它.
@omerhakanbilici由于性能原因,像`ArrayList`这样的实现会覆盖它.您指的是仅默认实现.

4> Ashish..:

由于问题已经得到解答,即最好的方法是使用迭代器对象的remove方法,我将进入"java.util.ConcurrentModificationException"抛出错误的地方的细节.

每个集合类都有它实现了Iterator接口,并提供了类似方法的私有类next(),remove()hasNext().

next的代码看起来像这样......

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

这里的方法checkForComodification实现为

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

因此,正如您所看到的,如果您明确尝试从集合中删除元素.它导致modCount与众不同expectedModCount,导致异常ConcurrentModificationException.



5> RodeoClown..:

你可以像你提到的那样直接使用迭代器,或者保留第二个集合并将要删除的每个项目添加到新集合中,然后在最后删除所有项目.这允许你继续使用for-each循环的类型安全性,代价是增加内存使用和cpu时间(除非你有真正的大型列表或真正的旧计算机,否则不应该是一个大问题)

public static void main(String[] args)
{
    Collection l = new ArrayList();
    Collection itemsToRemove = new ArrayList();
    for (int i=0; i < 10; ++i) {
    l.add(new Integer(4));
    l.add(new Integer(5));
    l.add(new Integer(6));
    }
    for (Integer i : l)
    {
        if (i.intValue() == 5)
            itemsToRemove.add(i);
    }

    l.removeAll(itemsToRemove);
    System.out.println(l);
}


这就是我通常所做的,但显式迭代器是我觉得更有利的解决方案.

6> Landei..:

在这种情况下,一个常见的伎俩(是?)倒退:

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

尽管如此,我很乐意,你必须在Java中8,如更好的方式removeIffilter在流.


这是一个很好的技巧.但是对于像集合这样的非索引集合它不会起作用,并且在链表上说它真的很慢.

7> Antzi..:

与Claudius相同的答案有一个for循环:

for (Iterator it = objects.iterator(); it.hasNext();) {
    Object object = it.next();
    if (test) {
        it.remove();
    }
}



8> Donald Raab..:

使用Eclipse Collections(以前称为GS Collections),MutableCollection上removeIf定义的方法将起作用:

MutableList list = Lists.mutable.of(1, 2, 3, 4, 5);
list.removeIf(Predicates.lessThan(3));
Assert.assertEquals(Lists.mutable.of(3, 4, 5), list);

使用Java 8 Lambda语法,可以编写如下:

MutableList list = Lists.mutable.of(1, 2, 3, 4, 5);
list.removeIf(Predicates.cast(integer -> integer < 3));
Assert.assertEquals(Lists.mutable.of(3, 4, 5), list);

Predicates.cast()这里需要调用,因为在Java 8中removeIfjava.util.Collection接口上添加了一个默认方法.

注意:我是Eclipse Collections的提交者.



9> Priyank Dosh..:

制作现有列表的副本并迭代新副本.

for (String str : new ArrayList(listOfStr))     
{
    listOfStr.remove(/* object reference or index */);
}


制作副本听起来像是浪费资源.
@Antzi这取决于列表的大小和内部对象的密度.仍然是有价值且有效的解决方案

10> Lluis Felisa..:

带有传统的for循环

ArrayList myArray = new ArrayList<>();

   for (int i = 0; i < myArray.size(); ) {
        String text = myArray.get(i);
        if (someCondition(text))
             myArray.remove(i);
        else 
             i++;
      }



11> John..:

人们断言一个人无法从被foreach循环迭代的Collection中删除.我只是想指出技术上是不正确的并准确描述(我知道OP的问题是如此先进以至于不知道这一点)这个假设背后的代码:

    for (TouchableObj obj : untouchedSet) {  // <--- This is where ConcurrentModificationException strikes
        if (obj.isTouched()) {
            untouchedSet.remove(obj);
            touchedSt.add(obj);
            break;  // this is key to avoiding returning to the foreach
        }
    }

并不是你不能从迭代中删除Colletion而是你不能再继续迭代.因此break在上面的代码中.

抱歉,如果这个答案是一个有点专业的用例,并且更适合我从这里来到这里的原始帖子,那个被标记为重复(尽管这个线程看起来更细致)并锁定.

推荐阅读
U友50081205_653
这个屌丝很懒,什么也没留下!
DevBox开发工具箱 | 专业的在线开发工具网站    京公网安备 11010802040832号  |  京ICP备19059560号-6
Copyright © 1998 - 2020 DevBox.CN. All Rights Reserved devBox.cn 开发工具箱 版权所有