边循环边删除List中的数据

List边循环,边删除;这种一听感觉就像是会出问题一样,其实只要是删除特定数据,就不会出问题,你如果直接循环删除所有数据,那可能就会出问题了,比如:

    public static void main(String[] args) {
        List<String> list = new ArrayList<>();
        list.add("1");
        list.add("2");
        list.add("3");
        list.add("4");

        for (int i=0;i<list.size();i++){
            list.remove(i);
        }
        
        System.out.println(list);

    }

结果:

[2, 4]

删除的所有,但是最终还留下两个值,这是因为当一个元素被移除时,该List的大小(size)就会缩减,同时也改变了索引的指向,也就是上面的代码只会循环两次,长度在不断减少,第一次循环0 < 4 ,第二次循环 1 < 3 ,不满足下一次循环条件 2 < 2,故只有两次循环就结束。所以,在迭代的过程中使用索引,将无法从List中正确地删除多个指定的元素。

当使用了foreach 如下:

    public static void main(String[] args) {
        List<String> list = new ArrayList<>();
        list.add("1");
        list.add("2");
        list.add("3");
        list.add("4");

        for (String i:list){
            list.remove(i);
        }
       
        System.out.println(list);

    }

结果直接报错:

Exception in thread "main" java.util.ConcurrentModificationException
	at java.util.ArrayList$Itr.checkForComodification(ArrayList.java:901)
	at java.util.ArrayList$Itr.next(ArrayList.java:851)
	at com.gwh.demo.test.ListTest.main(ListTest.java:18)

这是因为,在 foreach循环中,编译器使得 remove()方法先于next()方法被调用,因为先执行了remove()方法,导致next()获取的数组长度和remove()后的数组长度不一致,则抛出异常。

如果使用了Iterator 迭代器模式,如下:

    public static void main(String[] args) {
        List<String> list = new ArrayList<>();
        list.add("1");
        list.add("2");
        list.add("3");
        list.add("4");
        
        Iterator<String> iterator = list.iterator();
        while (iterator.hasNext()){
            // 必须先执行一下next() 否则会抛出异常
            iterator.next();
            iterator.remove();
        }
        
        System.out.println(list);

    }

这样保证next()先执行一下,这样删除所有就不会出问题了。

如果是循环List,同时删除符合要求的数据,则不管使用那种方式都不会出现问题。demo代码如下:

for循环:

    public static void main(String[] args) {
        HashMap<String, Object> map = new HashMap<String, Object>();
        map.put("id",1);
        map.put("name","张三");
        map.put("sex","男");
        map.put("age",18);

        HashMap<String, Object> map1 = new HashMap<String, Object>();
        map1.put("id",2);
        map1.put("name","李四");
        map1.put("sex","男");
        map1.put("age",18);

        HashMap<String, Object> map2 = new HashMap<String, Object>();
        map2.put("id",3);
        map2.put("name","王五");
        map2.put("sex","男");
        map2.put("age",18);

        HashMap<String, Object> map3 = new HashMap<String, Object>();
        map3.put("id",4);
        map3.put("name","赵六");
        map3.put("sex","男");
        map3.put("age",18);

        List<HashMap<String,Object>> list = new ArrayList<HashMap<String,Object>>();
        list.add(map);
        list.add(map1);
        list.add(map2);
        list.add(map3);
        
        for(int i=0;i<list.size();i++){
            if(Integer.valueOf(String.valueOf(list.get(i).get("id")))== 1){
                list.remove(i);
            }
        }

        System.out.println(list);
    }

结果如下:

[{sex=男, name=李四, id=2, age=18}, {sex=男, name=王五, id=3, age=18}, {sex=男, name=赵六, id=4, age=18}]

迭代器如下:

    public static void main(String[] args) {
        HashMap<String, Object> map = new HashMap<String, Object>();
        map.put("id",1);
        map.put("name","张三");
        map.put("sex","男");
        map.put("age",18);

        HashMap<String, Object> map1 = new HashMap<String, Object>();
        map1.put("id",2);
        map1.put("name","李四");
        map1.put("sex","男");
        map1.put("age",18);

        HashMap<String, Object> map2 = new HashMap<String, Object>();
        map2.put("id",3);
        map2.put("name","王五");
        map2.put("sex","男");
        map2.put("age",18);

        HashMap<String, Object> map3 = new HashMap<String, Object>();
        map3.put("id",4);
        map3.put("name","赵六");
        map3.put("sex","男");
        map3.put("age",18);
        
        List<HashMap<String,Object>> list = new ArrayList<HashMap<String,Object>>();
        list.add(map);
        list.add(map1);
        list.add(map2);
        list.add(map3);
        
        Iterator<HashMap<String, Object>> iterator = list.iterator();
        while (iterator.hasNext()){
            HashMap<String, Object> next = iterator.next();
            if(Integer.valueOf(String.valueOf(next.get("id")))== 1){
                iterator.remove();
            }
        }

        System.out.println(list);
        
    }

结果:

[{sex=男, name=李四, id=2, age=18}, {sex=男, name=王五, id=3, age=18}, {sex=男, name=赵六, id=4, age=18}]

还可以使用 JDK8特性stream 流过滤:

public static void main(String[] args) {
        HashMap<String, Object> map = new HashMap<String, Object>();
        map.put("id",1);
        map.put("name","张三");
        map.put("sex","男");
        map.put("age",18);

        HashMap<String, Object> map1 = new HashMap<String, Object>();
        map1.put("id",2);
        map1.put("name","李四");
        map1.put("sex","男");
        map1.put("age",18);

        HashMap<String, Object> map2 = new HashMap<String, Object>();
        map2.put("id",3);
        map2.put("name","王五");
        map2.put("sex","男");
        map2.put("age",18);

        HashMap<String, Object> map3 = new HashMap<String, Object>();
        map3.put("id",4);
        map3.put("name","赵六");
        map3.put("sex","男");
        map3.put("age",18);

        List<HashMap<String,Object>> list = new ArrayList<HashMap<String,Object>>();
        list.add(map);
        list.add(map1);
        list.add(map2);
        list.add(map3);

        List<HashMap<String, Object>> co = list.stream()
                .filter(p -> Integer.valueOf(String.valueOf(p.get("id"))) != 1)
                .collect(Collectors.toList());
        System.out.println(co);
    }

结果:

[{sex=男, name=李四, id=2, age=18}, {sex=男, name=王五, id=3, age=18}, {sex=男, name=赵六, id=4, age=18}]
相关推荐
遇见你真好。2 天前
springboot将文件处理成压缩文件
list·springboot
清酒伴风(面试准备中......)3 天前
Java集合 List——针对实习面试
java·面试·list·实习
兔兔爱学习兔兔爱学习3 天前
leetcode61. Rotate List
数据结构·list
hello world smile3 天前
Dart中List API用法大全
flutter·list·dart
Word的妈呀4 天前
Scala的List
开发语言·list·scala
魔道不误砍柴功4 天前
巧妙注入的奥秘:在 Spring 中优雅地使用 List 和 Map
数据库·spring·list
爱吃土豆的马铃薯ㅤㅤㅤㅤㅤㅤㅤㅤㅤ5 天前
两个list集合取重复数据
java·数据结构·windows·list
dawn6 天前
鸿蒙ArkTS中的布局容器组件(Scroll、List、Tabs)
华为·list·harmonyos·arkts·deveco studio·scroll·tabs
Kiki5296 天前
c++:list模拟实现
开发语言·c++·list
flying jiang6 天前
快速复制两个list
数据结构·windows·list