java cursor 删除_JAVA中循环删除(remove掉)list中元素的方法总结

平常我们在list循环中删除元素时 一不小心就会报java.util.ConcurrentModificationException 错误,那怎么样删除才不会报错呢

1.首先说下有几种循环:

1> for循环遍历list:for(inti=0;i

}

2> 增强for循环for(Stringx:list){

}

3> iterator遍历(迭代)Iteratorit=list.iterator();

while(it.hasNext()){

Stringx=it.next();

}

2. 3种循环在遍历集合时,对集合本身操作的注意点:

——for循环遍历list时,对集合进行add或remove操作for(inti=0;i

if(list.get(i).equals("del"))

list.remove(i);

}

—会报java.util.ConcurrentModificationException 异常

这种方式的问题在于,删除某个元素后,list的大小也发生了变化,而你的索引也在改变,所以会导致你在遍历的时候漏掉某些元素。比如当你删除第1个元素后,继续根据索引访问第2个元素时,因为删除的关系后面的元素都往前移动了一位,所以实际上访问的是第3个。因此,这种方式可以用在删除特定的一个元素时使用,但不适合循环删除多个元素的使用。

—解决方案:1>倒过来遍历list

for(inti=list.size()-1;i>=0;i--){

list.remove(i);

}

2>每移除一个元素以后再把i移过来

for(inti=0;i

list.remove(i)

i--;

}

——增强for循环和iterator遍历时,对集合进行add或remove操作for(Stringx:list){

list.remove(x);//list.add(x);

}

—iterator(迭代)遍历

Iteratorit=list.iterator();

while(it.hasNext()){

Stringx=it.next();

if(x.equals("del")){

list.remove();//应该用iterator自己的remove方法

}

}

—会报java.util.ConcurrentModificationException //这是一个并发修改异常报错

at java.util.AbstractList$Itr.checkForComodification(Unknown Source)

at java.util.AbstractList$Itr.next(Unknown Source)

异常

1>在遍历集合过程中,增强for循环不能对集合本身进行增删操作!

2>该list每当删除一个元素时,集合的size方法的值都会减小1,这将直接导致集合中元素的索引重新排序,

进一步说,就是剩余所有元素的索引值都减1,而for循环语句的局部变量i仍在递增,这将导致删除操作

发生跳跃。从而导致上述代码的删除出现问题。所以不要在增强for循环里进行元素的remove/add操作!

3>这种方式的问题在于,删除元素后继续循环会报错,因为元素在使用时发生了并发的修改,导致异常抛出。

但是删除完毕马上使用break跳出,则不会触发报错。

for(Stringx:list){

if(x.equals("del")){

list.remove(x);

break;

}

}

4>在遍历集合时,增强for的遍历是封装了迭代器的试遍历方式!(如下)

5>在AbstractList$Itr这个类中实现了iterator接口,当使用增强的for循环时,应该是使用迭代器

进行迭代了,如果你在这期间使用了add或remove方法的话,在ArrayList类中执行了这样的代码:

publicbooleanadd(E e){

ensureCapacity(size+1);// Increments(增长) modCount!!

elementData[size++]=e;

returntrue;

}

--add中的ensureCapacity(size+1);的调用代码:

publicvoidensureCapacity(intminCapacity){

modCount++;//modCount是AbstractList类中的一个成员变量,该值表示对List的修改次数!

intoldCapacity=elementData.length;

if(minCapacity>oldCapacity){

ObjectoldData[]=elementData;

intnewCapacity=(oldCapacity*3)/2+1;

if(newCapacity

newCapacity=minCapacity;

// minCapacity is usually close to size, so this is a win:

elementData=Arrays.copyOf(elementData,newCapacity);

}

}

--这里的modCount增加了,但是迭代器中的next方法时,

publicEnext(){

checkForComodification();

try{

Enext=get(cursor);//cursor:表示下一个要访问的元素的索引

lastRet=cursor++;// lastRet:表示上一个访问的元素的索引

returnnext;

}catch(IndexOutOfBoundsExceptione){

checkForComodification();

thrownewNoSuchElementException();

}

}

--先checkForComodification();

finalvoidcheckForComodification(){

if(modCount!=expectedModCount)//expectedModCount:表示对ArrayList修改次数的期望值,它的初始值为modCount为0。

thrownewConcurrentModificationException();

}

}

—因此抛出了异常。。。

——通俗点讲:

迭代器内部的每次遍历都会记录List内部的modcount当做预期值,然后在每次循环中用预期值与List

的成员变量modCount作比较,但是普通list.remove调用的是List的remove,这时modcount++,

但是iterator内记录的预期值=并没有变化,所以会报错,

—但是如果在iterator中调用remove,这时会同步List的modCount到iterator中,故不再报错。

Iteratorit=list.iterator();

while(it.hasNext()){

Stringx=it.next();

if(x.equals("del")){

it.remove();

}

}

—-再简单点说:

调用list.remove()方法导致modCount和expectedModCount的值不一致而报异常

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值