IT评测·应用市场-qidao123.com
标题:
集合之List--List如何安全删除
[打印本页]
作者:
钜形不锈钢水箱
时间:
2025-3-22 05:35
标题:
集合之List--List如何安全删除
在java中,List的删除操纵大概会导致ConcurrentModificationException一场,尤其是在遍历时直接调用removce(方法)
一、使用Iterator删除
Iterator提供了安全的删除方法remove(),可以在遍历时删除元素
List<String> list = new ArrayList<>();
list.add("A");
list.add("B");
list.add("C");
Iterator<String> iterator = list.iterator();
while (iterator.hasNext()) {
String item = iterator.next();
if (item.equals("B")) {
iterator.remove(); // 安全删除
}
}
System.out.println(list); // 输出: [A, C]
复制代码
优点:
安全且高效
适用于单线程环境
二、使用removeIf(方法)(java8+)
java8引入了removeIf方法,可以根据条件删除元素
List<String> list = new ArrayList<>();
list.add("A");
list.add("B");
list.add("C");
list.removeIf(item -> item.equals("B")); // 删除满足条件的元素
System.out.println(list); // 输出: [A, C]
复制代码
优点
:
代码简洁,得当函数式编程风格。
内部使用 Iterator,安全高效。
三、使用for循环倒序遍历
如果使用普通的for循环删除元素,建议倒序遍历,避免索引错误问题
List<String> list = new ArrayList<>();
list.add("A");
list.add("B");
list.add("C");
for (int i = list.size() - 1; i >= 0; i--) {
if (list.get(i).equals("B")) {
list.remove(i); // 删除元素
}
}
System.out.println(list); // 输出: [A, C]
复制代码
优点
:
避免索引错位问题。
适用于需要根据索引删除的场景。
四、使用CopyOnWriteArrayList
如果需要在多线程环境下安全删除元素,可以使用CopyOnWriteArrayList,它在修改时创建底层数据的新副本,得当多读写啊后的场景
List<String> list = new CopyOnWriteArrayList<>();
list.add("A");
list.add("B");
list.add("C");
for (String item : list) {
if (item.equals("B")) {
list.remove(item); // 安全删除
}
}
System.out.println(list); // 输出: [A, C]
复制代码
注意
:
CopyOnWriteArrayList 的写操纵性能较差,由于每次修改都会创建新副本。
得当多线程环境。
五、使用Collections.synchronizedList加锁
如果使用Collecions.synchronizedList包装List,需要手动加锁以保证线程安全
List<String> list = Collections.synchronizedList(new ArrayList<>());
list.add("A");
list.add("B");
list.add("C");
synchronized (list) {
Iterator<String> iterator = list.iterator();
while (iterator.hasNext()) {
String item = iterator.next();
if (item.equals("B")) {
iterator.remove(); // 安全删除
}
}
}
System.out.println(list); // 输出: [A, C]
复制代码
优点
:
线程安全。
适用于多线程环境
六、使用Stream过滤
如果需要删除满意条件的元素并生成新表,可以使用Stream的filter方法
List<String> list = new ArrayList<>();
list.add("A");
list.add("B");
list.add("C");
List<String> result = list.stream()
.filter(item -> !item.equals("B")) // 过滤掉 "B"
.collect(Collectors.toList());
System.out.println(result); // 输出: [A, C]
复制代码
优点
:
不修改原列表,生成新列表。
得当函数式编程风格。
总结
方法适用场景特点Iterator.remove()单线程环境安全高效removeIf()Java 8+,单线程环境简洁,得当函数式编程倒序遍历 for 循环根据索引删除避免索引错位CopyOnWriteArrayList多线程环境线程安全,得当读多写少Collections.synchronizedList + 锁多线程环境线程安全,需要手动加锁Stream.filter()生成新列表不修改原列表,得当函数式编程
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
欢迎光临 IT评测·应用市场-qidao123.com (https://dis.qidao123.com/)
Powered by Discuz! X3.4