在对List、Set、Map执行遍历删除或添加等改变集合个数的操作时,不能使用普通的while、for循环或增强for。会抛出ConcurrentModificationException异常或者没有达到删除的需求。在遍历时删除元素,需要使用迭代器的方式。

ArrayList源码中说明的报异常原因:

* <p>The iterators returned by this class's <tt>iterator</tt> and

 * <tt>listIterator</tt> methods are <i>fail-fast</i>: if the list is

 * structurally modified at any time after the iterator is created, in any way

 * except through the iterator's own <tt>remove</tt> or <tt>add</tt> methods,

 * the iterator will throw a {@link ConcurrentModificationException}.  Thus, in

 * the face of concurrent modification, the iterator fails quickly and cleanly,

 * rather than risking arbitrary, non-deterministic behavior at an undetermined

 * time in the future.<p>

(翻译:通过类的iterator和listiterator方法获取到的迭代器是快速失败迭代器:如果list在迭代器生成之后发生了结构性的改变,迭代器将抛出ConcurrentModificationException,但是当使用迭代器自己的remove或add方法时,不会抛出此异常。也就是说,当面对并发修改时,迭代器快速失败,而不是冒在未来不确定的时间发生不确定的行为的危险。)

 *

 * Note that the fail-fast behavior of an iterator cannot be guaranteed

 * as it is, generally speaking, impossible to make any hard guarantees in the

 * presence of unsynchronized concurrent modification.  Fail-fast iterators

 * throw <tt>ConcurrentModificationException</tt> on a best-effort basis.

 * Therefore, it would be wrong to write a program that depended on this

 * exception for its correctness: <i>the fail-fast behavior of iterators

 * should be used only to detect bugs.</i><p>

(翻译:需要注意的是迭代器不保证快速失败行为一定发生,因为一般来说不可能对是否发生了不同步并发修改做任何硬性的保证。快速失败迭代器会尽最大努力抛出ConcurrentModificationException异常。因此,写一个通过是否出现这种异常来判断是否正确的程序是错误的。快速失败行为的正确用法是仅用于检测异常。)

代码示例:

public class CollectionRemoveDemo {

    public static void main(String[] args) {
        ListRemove();
        System.out.println("-----------------------------------------------------------------------------------------------");
        SetRemove();
        System.out.println("-----------------------------------------------------------------------------------------------");
        MapRemove();
    }

    public static void ListRemove(){
        List<String> strList = new ArrayList<String>();
        strList.add("aaaa");
        strList.add("bbbb");
        strList.add("cccc");
        strList.add("cccc");
        strList.add("dddd");
        for(String str : strList){
            System.out.println(str);
        }
        System.out.println("init List size:" + strList.size());
        Iterator<String> it = strList.iterator();
        while(it.hasNext()){
            String str = it.next();
            if(str.equals("cccc")){
                it.remove();
            }
        }
        for(String str : strList){
            System.out.println(str);
        }
        System.out.println("removed List size:" + strList.size());
    }

    public static void SetRemove(){
        Set<String> strSet = new TreeSet<String>();
        strSet.add("aaaa");
        strSet.add("bbbb");
        strSet.add("cccc");
        strSet.add("cccc");//重复的数据将不会再次插入
        strSet.add("dddd");
        for(String str : strSet){
            System.out.println(str);
        }
        System.out.println("Init Set size:" + strSet.size());
        Iterator<String> it = strSet.iterator();
        while(it.hasNext()){
            String str = it.next();
            if(str.equals("cccc")){
                it.remove();
            }
        }
        for(String str : strSet){
            System.out.println(str);
        }
        System.out.println("removed Set size:" + strSet.size());
    }

    public static void MapRemove(){
        Map<String, String> strMap = new TreeMap<String, String>();
        strMap.put("a", "aaaa");
        strMap.put("b", "bbbb");
        strMap.put("c", "cccc");
        strMap.put("d", "dddd");
        for(String key : strMap.keySet()){
            System.out.println(key + " : " + strMap.get(key));
        }
        System.out.println("Init Map size:" + strMap.size());
        Iterator<Entry<String,String>> it = strMap.entrySet().iterator();
        while(it.hasNext()){
            Entry<String,String> strEntry = it.next();
            if(strEntry.getKey().equals("c")){
                it.remove();
            }
        }
        for(String key : strMap.keySet()){
            System.out.println(key + " : " + strMap.get(key));
        }
        System.out.println("removed Map size:" + strMap.size());
    }
}

Java遍历时删除List、Set、Map中的元素(源码分析)的更多相关文章

  1. BIZ中model.getSql源码分析

    功能:根据model.xml文件中配置的sql,获取对应的动态sql结果. 实例代码:String sql1 = model.getSql(dao.dbMeta());String sql2 = mo ...

  2. Spring中Bean命名源码分析

    Spring中Bean命名源码分析 一.案例代码 首先是demo的整体结构 其次是各个部分的代码,代码本身比较简单,不是我们关注的重点 配置类 /** * @Author Helius * @Crea ...

  3. HashSet 添加/遍历元素源码分析

    HashSet 类图 HashSet 简单说明 HashSet 实现了 Set 接口 HashSet 底层实际上是由 HashMap 实现的 public HashSet() { map = new ...

  4. Java并发编程笔记之Unsafe类和LockSupport类源码分析

    一.Unsafe类的源码分析 JDK的rt.jar包中的Unsafe类提供了硬件级别的原子操作,Unsafe里面的方法都是native方法,通过使用JNI的方式来访问本地C++实现库. rt.jar ...

  5. 【朝花夕拾】Android自定义View篇之(六)Android事件分发机制(中)从源码分析事件分发逻辑及经常遇到的一些“诡异”现象

    前言 转载请注明,转自[https://www.cnblogs.com/andy-songwei/p/11039252.html]谢谢! 在上一篇文章[[朝花夕拾]Android自定义View篇之(五 ...

  6. Netty中的ChannelPipeline源码分析

    ChannelPipeline在Netty中是用来处理请求的责任链,默认实现是DefaultChannelPipeline,其构造方法如下: private final Channel channel ...

  7. Springboot中注解@Configuration源码分析

    Springboot中注解@Configuration和@Component的区别 1.先说结论,@Configuration注解上面有@Component注解,所以@Component有的功能@Co ...

  8. java 集合与数组的互转方法,与源码分析

    前言 java数组与集合需要互相转换的场景非常多,但是运用不好还是容易抛出UnSupportedOperationException.下面讲解一下互转的方法,以及结合源码分异常产生的原因 集合转数组 ...

  9. MapReduce中map并行度优化及源码分析

    mapTask并行度的决定机制 一个job的map阶段并行度由客户端在提交job时决定,而客户端对map阶段并行度的规划的基本逻辑为:将待处理数据执行逻辑切片(即按照一个特定切片大小,将待处理数据划分 ...

随机推荐

  1. [BZOJ]4405: [wc2016]挑战NPC(带花树)

    带花树模板 #include<cstdio> #include<cstring> #include<algorithm> using namespace std; ...

  2. [LSGDOJ1822]书架 Splay

    题目描述 Sally有一个很大的书柜.这个书柜的构造有些独特,即书柜里的书是从上至下堆放成一列.她用1到n的正整数给每本书都编了号. Sally在看书的时候,每次取出一本书,看完后放回书柜然后再拿下一 ...

  3. hdu 4578 线段树(标记处理)

    Transformation Time Limit: 15000/8000 MS (Java/Others)    Memory Limit: 65535/65536 K (Java/Others) ...

  4. matplotlib中subplot的各参数的作用

    subplot(a,b,c)中a代表所画图形的行数 b代表所画图形的列数 c代表所画图形的序号. plt.figure(facecolor='w', figsize=(9, 10)) plt.subp ...

  5. C++多态实现原理

    C++的多态性用一句话概括就是:在基类的函数前加上virtual关键字,在派生类中重写该函数,运行时将会根据对象的实际类型来调用相应的函数.如果对象类型是派生类,就调用派生类的函数:如果对象类型是基类 ...

  6. [坑况]饿了么你是这样的前端——vue+element ui 【this dependency was not found:'element-ui/lib/theme-chalk/index.css'】

    element ui 坑况:今日pull代码,潇洒npm run dev ,被告知:this dependency was not found:'element-ui/lib/theme-chalk/ ...

  7. new File()

    首先 File 类是对文件系统的映射 并不是硬盘上真实的文件所以 new File("xxx.xxx") 只是在内存中创建File文件映射对象,而并不会在硬盘中创建文件 如果需要创 ...

  8. weblogic静默方式创建域

    创建域目录:mkdir -p /home/weblogic/Oracle/Middleware/user_projects/domains/base_domain/ 创建文件: create_doma ...

  9. Python练习之pillow

    此系列意在记录于一些有趣的程序及对其的总结. 问题来源: https://github.com/Yixiaohan/show-me-the-code https://github.com/HT524/ ...

  10. node安装教程

    推荐安装教程博客: https://www.cnblogs.com/zhouyu2017/p/6485265.html