在遍历List的过程中删除元素

首先遍历List的方式有以下几种:

  • 普通for循环
  • foreach循环
  • 使用迭代器(Iterator)
/**
     * 使用foeach循环 
     * 在循环过程中从List中删除元素以后,继续循环List时会抛出
     * ConcurrentModificationException 
     */
    public void listRemove() {
        List<Handler> handlers = this.handlers();
        for (Handler handler : handlers) {
            if (handler.getId() == 2)
                handlers.remove(handler);
        }
    }
    /** 
     * 像这种foreach循环+break操作对List进行遍历删除,但删除之后马上就跳出的也
     * 不会出现异常 
     */  
    public void listRemoveBreak() {  
        List<Handler> handlers = this.Handlers();  
        for (Handler handler : handlers) {  
            if (handler.getId() == 2) {  
                handlers.remove(handler);  
                break;  
            }  
        }  
    }  
    /** 
     * 这种遍历有可能会遗漏某个元素,因为删除元素后List的size在 
     * 变化,元素的索引也在变化,比如你循环到第2个元素的时候你把它删了, 
     * 接下来你去访问第3个元素,实际上访问到的是原先的第4个元素。当访问的元
     * 素 
     * 索引超过了当前的List的size后还会出现数组越界的异常,当然这里不会出现
     * 这种异常, 
     * 因为这里每遍历一次都重新拿了一次当前List的size。 
     */  
    public void listRemove2() {  
        List<handlers> handlers = this.getHandlers();  
        for (int i=0; i<handlers.size(); i++) {  
            if (handlers.get(i).getId() == 1) {  
                Handler handler = handlers.get(i);  
                handlers.remove(handler);  
            }  
        }  
    }  
    /** 
     * 使用Iterator的方式也可以顺利删除和遍历 
     */  
    public void iteratorRemove() {  
        List<Handlers> handlers = this.getHandlers();  
        System.out.println(handlers);  
        Iterator<Handler> handlIter = handlers.iterator();  
        while (handlIter.hasNext()) {  
            Handler handler = handlIter.next();  
            if (handler.getId()  == 1)  
                handlIter.remove();//这里要使用Iterator的remove方法移除当前对象,如果使用List的remove方法,则同样会出现ConcurrentModificationException  
        }  
        System.out.println(handlers);  
    }  

如果是在遍历操作远多于可变操作的时候,还可以可以考虑CopyOnWriteArrayList

    这一般需要很大的开销,但是当遍历操作的数量大大超过可变操作的数量时,这种方法可能比其他替代方法更 有效。在不能或不想进行同步遍历,但又需要从并发线程中排除冲突时,它也很有用。“快照”风格的迭代器方法在创建迭代器时使用了对数组状态的引用。此数组在迭代器的生存期内不会更改,因此不可能发生冲突,并且迭代器保证不会抛出ConcurrentModificationException。创建迭代器以后,迭代器就不会反映列表的添加、移除或者更改。在迭代器上进行的元素更改操作(remove、set和add)不受支持。这些方法将抛出UnsupportedOperationException。允许使用所有元素,包括null。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值