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

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

作者头像
Java技术栈
发布于 2023-03-08 06:47:14
发布于 2023-03-08 06:47:14
1.4K00
代码可运行
举报
文章被收录于专栏: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核心技术讲解--类集框架笔记
3.1、认识类集(理解) 如果现在要想保存多个对象,肯定使用对象数组完成,但是对象数组本身有一个最大的问题在于其数据的长度,所以后来使用了链表完成了动态对象数组的开发,可是链表的开发难度实在是很大,而且如果一个链表要想真正去使用,只依靠之前所编写的还不够,还需要进行一些代码的调优。 而在JDK 1.2之后正式引入了类集的概念,类集是一种动态的对象数组,属于各个数据结构的实现类,在整个类集之中主要的组成是一些核心的操作接口:Collection、List、Set、Map、Iterator、Enumeration。 3.2、单值保存的最大父接口:Collection(重点) 所谓的单值保存指的是每一次操作只会保存一个对象,就好像之前的链表程序一样,每一次只保存了一个对象,在Collection接口之中定义了如下的一些操作方法。 No. 方法名称 类型 描述 1 public boolean add(E e) 普通 数据增加 2 public void clear() 普通 清除数据 3 public boolean contains(Object o) 普通 查找数据是否存在 4 public boolean isEmpty() 普通 判断是否为空集合 5 public Iterator iterator() 普通 为Iterator接口实例化 6 public boolean remove(Object o) 普通 删除数据 7 public int size() 普通 取得集合的个数 8 public Object[] toArray() 普通 将集合变为对象数组 在Collection接口之中一共定义了15个方法,在所有的方法之中,只有两个方法最为常用:add()、iterator()。不过从开发上讲,很少会去直接使用Collection,都会使用Collection的两个子接口:List、Set。 3.3、允许重复的子接口:List(重点,80%) List是Collection的一个最为常用的子接口,首先这个接口的定义如下: public interface List extends Collection 但是List接口对Collection接口进行了大量的扩充,但是扩充之后的主要方法: No. 方法名称 类型 描述 1 public E get(int index) 普通 取得指定索引位置上的数据 2 public E set(int index, E element) 普通 修改指定索引位置上的数据 3 public ListIterator listIterator() 普通 为ListIterator接口实例化 但是以上的三个方法,只是针对于List接口起作用,而List接口有两个常用子类:ArrayList、Vector。 3.3.1、新的子类:ArrayList,95% ArrayList是List子接口使用最多的一个子类,而这个类的定义如下: public class ArrayList extends AbstractList implements List, RandomAccess, Cloneable, Serializable 按照面向对象的概念来讲,现在使用ArrayList主要的目的是为List接口实例化,所有的操作方法都以List接口为主。 范例:使用ArrayList进行List接口的功能验证 package cn.mldn.demo; import java.util.ArrayList; import java.util.List; public class TestDemo { public static void main(String[] args) throws Exception { List all = new ArrayList() ; all.add(“Hello”) ; all.add(“Hello”) ; // 内容重复了 all.add(“World”) ; for (int x = 0; x < all.size(); x++) { String str = all.get(x) ; // get()方法只有List接口有 System.out.print(str + “、”); } } } 在使用代码的时候可以发现,List集合之中即使存在了重复数据,也可以正常的保存,而且数据保存的顺序就是存入数据的顺序。 范例:使用List集合修改之前的程序 package cn.mldn.demo; import java.util.ArrayList; import java.util.List; interfac
葆宁
2019/04/18
6300
使用 `removeIf` 轻松移除集合元素:避免 `ConcurrentModificationException` 异常的最佳实践
在现代 Java 开发中,我们经常需要对集合进行操作,比如移除符合特定条件的元素。Java 8 引入了许多强大的新特性,其中之一就是 removeIf 方法。本文将通过一个实际的例子,详细介绍 removeIf 方法的背景、使用方法、注意事项,并举一些日常开发中常见的应用场景,同时也会展示如何避免 ConcurrentModificationException 异常。
九转成圣
2024/08/09
2840
Java进阶-集合(2)
Map是一种键-值对(key-value)集合, 集合中的每一个元素都包含一个键(key)对象和一个值(value)对象。用于保存具有映射关系的数据(Map是一种映射表,可以通过key快速查找value)。
reload
2024/02/28
1980
Java进阶-集合(2)
java---集合(数据结构)(重点)
以前存储一组相同类型的数据使用数组,固定大小,具有连续性的存储空间。比如,5个长度的数组再存入数据时,如果现在已经存满,存入第六个元素,这时数组空间不够,扩容。Arrays.copyOf() , 很不方便,如果扩容频率太高,也影响你程序运行效率。集合来解决数组固定,如果扩容又影响效率的问题
用户10787181
2023/10/17
2530
java---集合(数据结构)(重点)
Java中遍历HashMap的5种方式
HashMap是Java中最常用的集合之一,它实现了Map接口并提供了键值对的映射。在Java中,HashMap是一个非同步的类,它的主要目的是为了快速的数据访问和搜索。
王也518
2024/04/17
2880
Java中遍历HashMap的5种方式
有序的四字成语_LinkedHashMap
大家好,又见面了,我是你们的朋友全栈君。 HashMap是无序的,HashMap在put的时候是根据key的hashcode进行hash然后放入对应的地方。所以在按照一定顺序put进HashMap中,然后遍历出HashMap的顺序跟put的顺序不同(除非在put的时候key已经按照hashcode排序号了,这种几率非常小) 单纯的HashMap是无法实现排序的,这的排序是指,我们将键值对按照一定的顺序put进HashMap里,然后在进行取键值对的操作的时候,是按照put进去的顺序把键值对取出来的。
全栈程序员站长
2022/11/09
2880
java_Collection、Map、泛型的使用
集合按照其存储结构可以分为两大类,分别是 单列集合 java.util.Collection 双列集合 java.util.Map
咕咕星
2020/08/19
1.2K0
java_Collection、Map、泛型的使用
面试官:HashMap 为什么不能一边遍历一遍删除
上面出现这样的原因是在使用 foreach 对 HashMap 进行遍历时,同时进行 put 赋值操作会有问题,异常 ConcurrentModificationException。
路人甲Java
2023/08/29
3490
面试官:HashMap 为什么不能一边遍历一遍删除
面试官:怎么去除 List 中的重复元素?我一行代码搞定,赶紧拿去用!
上次栈长给大家分享了《带了一个 3 年的开发,不会循环删除 List 中的元素,我简直崩溃!!》,上次也给大家留了个小话题:
Java技术栈
2023/02/27
1.1K0
面试官:怎么去除 List 中的重复元素?我一行代码搞定,赶紧拿去用!
阶段01Java基础day18集合框架04
声明:本文为原创,作者为 对弈,转载时请保留本声明及附带文章链接:http://www.duiyi.xyz/c%e5%ae%9e%e7%8e%b0%e9%9b%b7%e9%9c%86%e6%88%98%e6%9c%ba-25/
对弈
2019/09/04
5420
Java HashMap遍历:KeySet、EntrySet、迭代器与Lambda表达式
在 Java 编程中,HashMap 是常用数据结构,用于存储键值对。实际开发中,高效灵活遍历是核心技巧。本文将深入探讨四种遍历 HashMap 的方法(基于 KeySet、EntrySet、Iterator、Lambda 表达式)并对比差异和适用场景
Yeats_Liao
2024/12/30
2520
Java HashMap遍历:KeySet、EntrySet、迭代器与Lambda表达式
HashMap 的 7 种遍历方式与性能分析!(强烈推荐)
随着 JDK 1.8 Streams API 的发布,使得 HashMap 拥有了更多的遍历的方式,但应该选择那种遍历方式?反而成了一个问题。
磊哥
2020/05/04
12.4K0
JavaSE集合(八)之Map
前面给大家介绍了集合家族中的Collection家族,这一篇给大家分享的是集合中的另一个家族就是Map家族。以前的时候学习Map的时候没有很认真的去学习,我觉得很多东西还是不是很清楚。 这次我将总结的非常详细。程序员的道理里,我们风雨无阻! 一、Map接口 1.1、Map概述   Map 的字面翻译是映射(地图就是一种映射)。将键映射到值的对象,一个映射不能包含重复的键(如果有添加有重复的键,后面的会覆盖掉前面的,但是如果是自定义类型必须重写hashCode()和equals()方法),每个键最多只能映射到
用户1195962
2018/01/18
9670
JavaSE集合(八)之Map
HashMap 和 Hashtable 的 6 个区别,最后一个没几个人知道!
HashMap 和 Hashtable 是 Java 开发程序员必须要掌握的,也是在各种 Java 面试场合中必须会问到的。
Java技术栈
2018/12/24
4920
HashMap 和 Hashtable 的 6 个区别,最后一个没几个人知道!
面试官上来就让手撕HashMap的7种遍历方式,当场愣住,最后只写出了3种
今天有个小伙伴私信诉苦,说面试官上来就让他手撕HashMap的7种遍历方式,最终只写出3种常用的,怀疑面试官是在故意刁难。这个问题大家怎么看?
JavaBuild
2024/05/27
1180
面试官上来就让手撕HashMap的7种遍历方式,当场愣住,最后只写出了3种
Java之HashMap迭代删除使用方法小结
map的迭代删除,和我们常见的list,set不太一样,不能直接获取Iteraotr对象,提供的删除方法也是单个的,根据key进行删除,如果我们有个需求,将map中满足某些条件的元素删除掉,要怎么做呢?
一灰灰blog
2019/07/02
1.5K0
Java——类集框架:Map集合的详解及应用举例(HashMap、Hashtable、TreeMap)
Map保存的是二元偶对象,简单说就是两个值,key和value。使用中可以通过key查找到value数据,使用Map可以方便用户查询。
Winter_world
2022/05/08
4210
Java——类集框架:Map集合的详解及应用举例(HashMap、Hashtable、TreeMap)
还在自己写迭代器进行remove?快来看看新方法
我们都知道 List 中是不允许在循环的过程中去进行移除元素的,为什么呢?一般的新人可能会遇到这个问题,比如说会从 List 的遍历的过程中去进行 remove 数据,但是干过几年的开发的有经验的工作人员,是肯定不会这么干的,很简单,会报错。
Java极客技术
2022/12/04
2510
还在自己写迭代器进行remove?快来看看新方法
增强for循环
jdk1.5出现的新特性---->增强for循环
MonroeCode
2018/01/11
7740
看完这篇 HashMap ,和面试官扯皮就没问题了
「如果你没有时间细抠本文,可以直接看 HashMap 概述,能让你对 HashMap 有个大致的了解」。
cxuan
2020/06/28
5660
看完这篇 HashMap ,和面试官扯皮就没问题了
相关推荐
李兴华Java核心技术讲解--类集框架笔记
更多 >
领券
💥开发者 MCP广场重磅上线!
精选全网热门MCP server,让你的AI更好用 🚀
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验