Loading [MathJax]/jax/output/CommonHTML/config.js
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >面试官:怎么删除 HashMap 中的元素?我一行代码搞定,赶紧拿去用!

面试官:怎么删除 HashMap 中的元素?我一行代码搞定,赶紧拿去用!

作者头像
Java技术栈
发布于 2023-03-08 06:47:14
发布于 2023-03-08 06:47:14
1.5K00
代码可运行
举报
文章被收录于专栏:Java技术栈Java技术栈
运行总次数:0
代码可运行

背景

大家好,我是栈长。

前些天,栈长给大家分享了两篇有意思的文章:

带了一个 3 年的开发,不会循环删除 List 中的元素,我简直崩溃!!

面试官:怎么去除 List 中的重复元素?我一行代码搞定,赶紧拿去用!

这两篇文章确实能帮助一大部分人,其中分享的一些实现技巧,编程很多年的高手也不一定用过,不管自己水平多牛,还是多谦虚好学一些,掌握多一点总不是什么坏事。

有粉丝建议栈长出一篇删除 HashMap 里面的数据,也有粉丝建议出一个系列的文章:

那这篇就分享下如何删除 HashMap 中的元素吧!

PS: 这仅是我个人掌握的实现方案,不一定全,也不一定是最优的,欢迎大家分享,杠精勿扰。

HashMap 删除元素方案

假设有以下数据:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public Map<String, String> initMap = new HashMap<>() {{
    put("user1", "张三");
    put("user2", "李四");
    put("user3", "张三");
    put("user4", "李四");
    put("user5", "王五");
    put("user6", "赵六");
    put("user7", "李四");
    put("user8", "王五");
}};

本文所有完整示例源代码已经上传:

https://github.com/javastacks/javastack

欢迎 Star 学习,后面 Java 示例都会在这上面提供!

一般删除 HashMap 集合中的元素,如果知道具体的 Key,并且需要根据 Key 删除元素,使用 remove 方法就可以了。但是如何根据 Value 删除 HashMap 集合中的元素呢?这才是你必须掌握的技巧!

1、使用 for 循环删除
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
/**
 * 使用 for 循环删除
 * @author: 栈长
 * @from: 公众号Java技术栈
 */
@Test
public void remove1() {
    Set<Map.Entry<String, String>> entries = new CopyOnWriteArraySet<>(initMap.entrySet());
    for (Map.Entry<String, String> entry : entries) {
        if ("张三".equals(entry.getValue())) {
            initMap.remove(entry.getKey());
        }
    }
    System.out.println(initMap);
}

输出结果:

{user2=李四, user7=李四, user8=王五, user5=王五, user6=赵六, user4=李四}

使用 HashMap 中实现的 entrySet 方法获取元素的集合,然后再进行循环遍历,先根据 Value 值判断要删除的元素,然后再根据 Key 删除元素。

在之前的文章中知道,增强的 for 循环底层使用的迭代器 Iterator,而 HashMap 是 fail-fast 类型的错误机制,所以遍历时删除元素会出现 java.util.ConcurrentModificationException 并发修改异常。

所以,这里使用了线程安全的 CopyOnWriteArraySet 封装了一层,避免出现并发修改异常,java.util.concurrent 包中的并发集合类都被设计为 fail-safe(安全失败)类型的,比如 CopyOnWrite* 、ConcerrentHashMap 集合,遍历过程中结构发生变更是安全的,不会抛出以上异常。

需要注意的是: 虽然 CopyOnWriteArraySet 并发性能很好,但每次删除时都会复制一份同等集合,所以要考虑数据过多可能导致的内存消耗问题。具体使用和实现原理可以点击该 CopyOnWriteArraySet 关键字链接看之前的文章,这里不再撰述。

2、使用 forEach 循环删除
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
/**
 * 使用 forEach 循环删除
 * @author: 栈长
 * @from: 公众号Java技术栈
 */
@Test
public void remove2() {
    ConcurrentHashMap<String, String> map = new ConcurrentHashMap<>(initMap);
    map.forEach((k, v) -> {
        if ("张三".equals(v)) {
            map.remove(k);
        }
    });
    System.out.println(map);
}

输出结果:

{user2=李四, user7=李四, user8=王五, user5=王五, user6=赵六, user4=李四}

使用 HashMap 自带的 forEach 循环删除指定值的元素,这里为什么使用线程安全的 ConcurrentHashMap 集合包装了一层,同样是为了避免并发修改异常。ConcurrentHashMap 在各版本中都使用了最优的锁设计方案,它的并发性能也是非常优异的。

另外,HashMap 和 ConcurrentHashMap 也是面试必问的,如果你近期准备面试跳槽,建议在Java面试库小程序在线刷题,涵盖 2000+ 道 Java 面试题,几乎覆盖了所有主流技术面试题。

3、使用 Iterator 迭代器删除
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
/**
 * 使用 Iterator 迭代器删除
 * @author: 栈长
 * @from: 公众号Java技术栈
 */
@Test
public void remove3() {
    Iterator<Map.Entry<String, String>> iterator = initMap.entrySet().iterator();
    while (iterator.hasNext()) {
        Map.Entry<String, String> entry = iterator.next();
        if ("张三".equals(entry.getValue())) {
            iterator.remove();
        }
    }
    System.out.println(initMap);
}

输出结果:

{user2=李四, user7=李四, user8=王五, user5=王五, user6=赵六, user4=李四}

这种方式即正常使用迭代器遍历删除,它不会发生并发修改异常。

需要注意的是: 这种方法虽然不会发生并发修改异常,但 HashMap 并不是线程安全的,在迭代删除元素时,另一个线程可能会删除 HashMap 中的数据, 这时使用迭代器删除同样会导致并发修改异常。 所以,要保证线程安全的删除,在创建迭代器之前,可以先用线程安全的 ConcurrentHashMap 集合包装一层。或者使用 synchronized 关键字锁住整个 Map。 如果没有多线程修改环境,可以不用考虑。

4、使用 removeIf 删除
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
/**
 * 使用 removeIf 删除
 * @author: 栈长
 * @from: 公众号Java技术栈
 */
@Test
public void remove4() {
    initMap.entrySet().removeIf(entry -> "张三".equals(entry.getValue()));
    System.out.println(initMap);
}

输出结果:

{user2=李四, user7=李四, user8=王五, user5=王五, user6=赵六, user4=李四}

使用 entrySet 的 removeIf 删除,它底层使用的是迭代器:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
default boolean removeIf(Predicate<? super E> filter) {
    Objects.requireNonNull(filter);
    boolean removed = false;
    final Iterator<E> each = iterator();
    while (each.hasNext()) {
        if (filter.test(each.next())) {
            each.remove();
            removed = true;
        }
    }
    return removed;
}

所以,它和方法 3 是一样的,只不过把条件写成了 Predicate 函数式接口而已。

需要注意的是: removeIf 虽然更方便了,但它仍然不是线程安全的,多线程场景参考方案同方法 3。

5、使用 Stream 删除
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
/**
 * 使用 Stream 删除
 * @author: 栈长
 * @from: 公众号Java技术栈
 */
@Test
public void remove5() {
    Map<String, String> map = initMap.entrySet().stream()
            .filter(entry -> !"张三".equals(entry.getValue()))
            .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue));
    System.out.println(map);
}

输出结果:

{user2=李四, user7=李四, user8=王五, user5=王五, user6=赵六, user4=李四}

利用 Stream 的 filter 方法进行过滤,这个方法也十分简单,一行代码搞定。Stream 基础就不介绍了,Stream 系列我之前写过一个专题了。

本文所有完整示例源代码已经上传:

https://github.com/javastacks/javastack

欢迎 Star 学习,后面 Java 示例都会在这上面提供!

总结

本文总结了 5 种删除 HashMap 元素的方法:

  • 使用 for 循环删除
  • 使用 forEach 循环删除
  • 使用 Iterator 迭代器删除
  • 使用 removeIf 删除
  • 使用 Stream 删除

实际开发过程中,可能会使用不同的遍历方式,所以重点要考虑多线程场景,如果只是简单的删除元素,使用 removeIf 和 Stream 过滤是最省事的。

所以说,你身边还有谁不会删除 HashMap 中的元素?把这篇文章发给他吧,让大家少走弯路,少写垃圾代码,共同进步。

你还知道哪些删除技巧?欢迎留言分享~

最后,留个话题:

上面的种种方法虽然能删除 HashMap 中指定值的元素,但是不能删除所有的重复元素,你觉得怎么删除重复数据比较好?有哪些方案?

大家可以先讨论下方案,下期分享,等栈长写完~

版权声明: 本文系公众号 "Java技术栈" 原创,转载、引用本文内容请注明出处,抄袭、洗稿一律投诉侵权,后果自负,并保留追究其法律责任的权利。

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2023-02-27,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 Java技术栈 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
Java中遍历HashMap的5种方式
HashMap是Java中最常用的集合之一,它实现了Map接口并提供了键值对的映射。在Java中,HashMap是一个非同步的类,它的主要目的是为了快速的数据访问和搜索。
王也518
2024/04/17
3440
Java中遍历HashMap的5种方式
面试官:怎么去除 List 中的重复元素?我一行代码搞定,赶紧拿去用!
上次栈长给大家分享了《带了一个 3 年的开发,不会循环删除 List 中的元素,我简直崩溃!!》,上次也给大家留了个小话题:
Java技术栈
2023/02/27
1.2K0
面试官:怎么去除 List 中的重复元素?我一行代码搞定,赶紧拿去用!
使用 `removeIf` 轻松移除集合元素:避免 `ConcurrentModificationException` 异常的最佳实践
在现代 Java 开发中,我们经常需要对集合进行操作,比如移除符合特定条件的元素。Java 8 引入了许多强大的新特性,其中之一就是 removeIf 方法。本文将通过一个实际的例子,详细介绍 removeIf 方法的背景、使用方法、注意事项,并举一些日常开发中常见的应用场景,同时也会展示如何避免 ConcurrentModificationException 异常。
九转成圣
2024/08/09
3580
Lambda【1】-- List相关Lambda表达式使用(上篇)
Lambda在jdk1.8里面已经很好用了,在这里不讲底层的实现,只有简单的用法,会继续补全。
秦怀杂货店
2020/12/26
9640
java陷阱之:HashMap for each遍历同时删除,抛出ConcurrentModificationException
当HashMap使用for each遍历entrySet的同时,使用HashMap的remove操作元素时,并不是在并发的情况下,也会抛出异常:ConcurrentModificationException。
崔认知
2023/06/19
5720
java陷阱之:HashMap for each遍历同时删除,抛出ConcurrentModificationException
Java进阶-集合(2)
Map是一种键-值对(key-value)集合, 集合中的每一个元素都包含一个键(key)对象和一个值(value)对象。用于保存具有映射关系的数据(Map是一种映射表,可以通过key快速查找value)。
reload
2024/02/28
2440
Java进阶-集合(2)
面试官:HashMap 为什么不能一边遍历一遍删除
上面出现这样的原因是在使用 foreach 对 HashMap 进行遍历时,同时进行 put 赋值操作会有问题,异常 ConcurrentModificationException。
路人甲Java
2023/08/29
3750
面试官:HashMap 为什么不能一边遍历一遍删除
还在自己写迭代器进行remove?快来看看新方法
我们都知道 List 中是不允许在循环的过程中去进行移除元素的,为什么呢?一般的新人可能会遇到这个问题,比如说会从 List 的遍历的过程中去进行 remove 数据,但是干过几年的开发的有经验的工作人员,是肯定不会这么干的,很简单,会报错。
Java极客技术
2022/12/04
2610
还在自己写迭代器进行remove?快来看看新方法
HashMap 的 7 种遍历方式
本文先从 HashMap 的遍历方法讲起,然后再从性能、原理以及安全性等方面,来分析 HashMap 各种遍历方式的优势与不足,本文主要内容如下图所示:
Vincent-yuan
2021/08/23
4720
JDK 11 及以上版本 Java 最新技术之集合框架实操应用解析
以下是基于Java最新技术(JDK 11+)的集合框架实操内容,结合了Java 8+的新特性(如Stream API、Lambda表达式)和并发编程的最佳实践。
啦啦啦191
2025/06/21
660
JDK 11 及以上版本 Java 最新技术之集合框架实操应用解析
java_Collection、Map、泛型的使用
集合按照其存储结构可以分为两大类,分别是 单列集合 java.util.Collection 双列集合 java.util.Map
咕咕星
2020/08/19
1.2K0
java_Collection、Map、泛型的使用
JavaSE集合(八)之Map
前面给大家介绍了集合家族中的Collection家族,这一篇给大家分享的是集合中的另一个家族就是Map家族。以前的时候学习Map的时候没有很认真的去学习,我觉得很多东西还是不是很清楚。 这次我将总结的非常详细。程序员的道理里,我们风雨无阻! 一、Map接口 1.1、Map概述   Map 的字面翻译是映射(地图就是一种映射)。将键映射到值的对象,一个映射不能包含重复的键(如果有添加有重复的键,后面的会覆盖掉前面的,但是如果是自定义类型必须重写hashCode()和equals()方法),每个键最多只能映射到
用户1195962
2018/01/18
9980
JavaSE集合(八)之Map
增强for循环
jdk1.5出现的新特性---->增强for循环
MonroeCode
2018/01/11
7900
有序的四字成语_LinkedHashMap
大家好,又见面了,我是你们的朋友全栈君。 HashMap是无序的,HashMap在put的时候是根据key的hashcode进行hash然后放入对应的地方。所以在按照一定顺序put进HashMap中,然后遍历出HashMap的顺序跟put的顺序不同(除非在put的时候key已经按照hashcode排序号了,这种几率非常小) 单纯的HashMap是无法实现排序的,这的排序是指,我们将键值对按照一定的顺序put进HashMap里,然后在进行取键值对的操作的时候,是按照put进去的顺序把键值对取出来的。
全栈程序员站长
2022/11/09
3000
HashMap有几种遍历方法?推荐使用哪种?
HashMap 的遍历方法有很多种,不同的 JDK 版本有不同的写法,其中 JDK 8 就提供了 3 种 HashMap 的遍历方法,并且一举打破了之前遍历方法“很臃肿”的尴尬。
磊哥
2021/12/09
1.2K0
HashMap有几种遍历方法?推荐使用哪种?
Map集合
摘要总结:本篇文章主要介绍了如何使用HashMap和TreeMap来存储和查找数据,以及如何使用Iterator迭代器进行迭代遍历。
片刻
2018/01/05
9130
Map、斗地主案例
发布者:全栈程序员栈长,转载请注明出处:https://javaforall.cn/100193.html原文链接:https://javaforall.cn
全栈程序员站长
2022/06/29
1.4K0
Map、斗地主案例
java hashmap 遍历删除元素_java 中 HashMap 遍历与删除[通俗易懂]
/*** 在键值都需要时使用*/Map map = new HashMap();for (Map.Entryentry : map.entrySet()) {
全栈程序员站长
2022/10/02
2.8K0
【Java】java 集合框架(详解)
🔥 Java集合框架 提供了一系列用于存储和操作对象组的接口和类。这些工具是为了解决不同数据结构通用操作的需求而设计的。集合框架主要包括两种类型的容器:
IsLand1314
2025/06/02
1640
【Java】java 集合框架(详解)
Java之HashMap迭代删除使用方法小结
map的迭代删除,和我们常见的list,set不太一样,不能直接获取Iteraotr对象,提供的删除方法也是单个的,根据key进行删除,如果我们有个需求,将map中满足某些条件的元素删除掉,要怎么做呢?
一灰灰blog
2019/07/02
1.5K0
相关推荐
Java中遍历HashMap的5种方式
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验