List的remove()方法详解

https://blog.csdn.net/anxin_hw/article/details/128312846

一、错误使用场景

1、普通for循环遍历List删除指定元素,list.remove(index)

示例:将姓张的名字移除掉

List<String> nameList = new ArrayList<>(Arrays.asList("张三", "李四", "王五", "赵六"));
nameList.add("张七");
nameList.add("张八");
for (int i = 0; i < nameList.size(); i++) {
     String name = nameList.get(i);
     System.out.println("当前遍历对象==="+name);
     if (name.startsWith("张")) {
         nameList.remove(i);
     }
}
System.out.println(nameList);

   输出结果:

 【李四】【张八】遍历被跳过,【张八】没有被成功移除

原因分析:

List调用remove(index)方法后,会移除index位置上的元素,之后的所有元素依次前移,当移除完【张三】时,【李四】变成了数组的第一位,此时的索引应该要index-1才能获取到【李四】。所以每移除一个元素时就需要把index-1,否则原来索引为index+1的元素就无法遍历到。

2、foreach遍历List删除元素

示例:同上

List<String> nameList = new ArrayList<>(Arrays.asList("张三", "李四", "王五", "赵六"));
nameList.add("张七");
nameList.add("张八");
for (String name : nameList) {
    System.out.println("当前遍历对象==="+name);
    if (name.startsWith("张")) {
        nameList.remove(name);
    }
}
System.out.println(nameList);

  输出结果:

 

原因分析:

foreach循环实际上是迭代,使用list.remove(item)方法后,list 对象的modCount值进行了修改,而 list对象的迭代器中的expectedModCount值没有修改,所以抛出了异常

3、直接使用list.remove(object)

示例: 将张三从数组中移除

List<String> nameList = new ArrayList<>(Arrays.asList("张三", "李四", "王五", "赵六"));
nameList.add("张三");
nameList.add("张三");
nameList.remove("张三");
System.out.println(nameList);

  

输出结果:[李四, 王五, 赵六, 张三, 张三]

原因分析:

list.remove(object)的逻辑和list.remove(index)大致相同:元素依次遍历,数组中存在与入参对象相等的元素就移除,之后的所有元素依次前移,返回true,若不存在与入参相等的元素,返回false。

二、正确使用方法

1、 使用list.removeIf()方法

removeIf()的入参是一个过滤条件,用来判断需要移除的元素是否满足条件。

原理:方法中设置了一个removeSet,把满足条件的元素索引坐标都放入removeSet,然后统一对removeSet中的索引进行移除

示例:将姓张的名字移除掉

List<String> nameList = new ArrayList<>(Arrays.asList("张三", "李四", "王五", "赵六"));
nameList.add("张七");
nameList.add("张八");
nameList.removeIf(o->o.startsWith("张"));
System.out.println(nameList);

  输出结果:[李四, 王五, 赵六]

2、使用迭代器Iterator移除元素

迭代器是一个链表,直接使用remove操作不会出现问题

示例:同上

List<String> nameList = new ArrayList<>(Arrays.asList("张三", "李四", "王五", "赵六"));
nameList.add("张七");
nameList.add("张八");
Iterator<String> iterator = nameList.iterator();
while (iterator.hasNext()) {
    if (iterator.next().startsWith("张")) {
        iterator.remove();
    }
}
System.out.println(nameList);

  输出结果:[李四, 王五, 赵六]

3、使用list.removeAll()方法

定义一个需要移除的数组,for循环内将需要移除的元素收集在定义的数组中,for循环结束后直接removeAll()

示例:同上

List<String> nameList = new ArrayList<>(Arrays.asList("张三", "李四", "王五", "赵六"));
nameList.add("张七");
nameList.add("张八");
List<String> removeList = new ArrayList<>();
for (String name : nameList) {
    if (name.startsWith("张")) {
        removeList.add(name);
    }
}
nameList.removeAll(removeList);
System.out.println(nameList);

  输出结果:[李四, 王五, 赵六]

4、倒序遍历元素然后删除

正序删除元素会移位那我们转变思维倒序遍历

示例:

List<String> nameList = new ArrayList<>(Arrays.asList("张三", "李四", "王五", "赵六"));
nameList.add("张七");
nameList.add("张八");
for (int i = nameList.size() - 1; i >= 0; i--) {
    String name = nameList.get(i);
    System.out.println("当前遍历对象===" + name);
    if (name.startsWith("张")) {
        nameList.remove(i);
    }
}
System.out.println(nameList);

  输出结果:

 

三、总结

1、使用普通for循环删除元素时需要注意索引会变化的问题

2、list删除元素时建议使用迭代器Iterator或者removeIf()方法

posted @ 2024-05-04 21:45  yinghualeihenmei  阅读(1147)  评论(0编辑  收藏  举报