JDK1.8 之前 HashMap 底层是 数组和链表, 之后在之前基础上加上红黑树。

相比于之前的版本, JDK1.8 之后在解决哈希冲突时有了较大的变化,当链表长度大于阈值(默认为 8)(将链表转换成红黑树前会判断,如果当前数组的长度小于 64,那么会选择先进行数组扩容,而不是转换为红黑树)时,将链表转化为红黑树,以减少搜索时间。

HashMap 在容量不够进行 resize 时由于高并发可能出现死链,导致 CPU 飙升,为了避免这种情况的发生,建议在高并发场景下,可以使用其他数据结构来替代 HashMap,比如 ConcurrentHashMap,它是一种线程安全的哈希表实现,在高并发情况下能够保证并发性能和数据一致性。另外,还可以通过加锁的方式来避免死链的发生,比如使用 ReentrantLock 等锁机制,来保证对 HashMap 的操作是线程安全的。

HashMap 在高并发场景下可能出现的性能问题以及如何规避这些问题

首先,HashMap 是一种常用的哈希表实现,它用于存储 key-value 键值对,并提供了快速的插入、删除和查找操作。在 HashMap 中,元素的存储位置是根据 key 的哈希值来计算的,相同哈希值的元素会被放置在同一个桶(bucket)中。当 HashMap 中的元素个数达到了负载因子(load factor)所设置的阈值时,就需要对 HashMap 进行扩容操作,从而保证 HashMap 的性能。

但是,在进行扩容操作时,由于需要重新计算元素的哈希值,重新放置元素的位置,以及调整桶的大小等操作,可能会导致多个线程同时对同一个桶进行操作,从而发生死链(链表死循环)的情况。当发生死链时,CPU 的使用率会急剧飙升,严重影响系统的性能。

为了避免这种情况的发生,建议在高并发场景下,可以使用其他数据结构来替代 HashMap,比如 ConcurrentHashMap,它是一种线程安全的哈希表实现,在高并发情况下能够保证并发性能和数据一致性。另外,还可以通过加锁的方式来避免死链的发生,比如使用 ReentrantLock 等锁机制,来保证对 HashMap 的操作是线程安全的。

综上所述,为了避免 HashMap 在高并发情况下出现死链的情况,我们可以选择使用其他数据结构,或者采取加锁等措施来保证对 HashMap 的操作是线程安全的,从而避免CPU飙升的情况。

死链问题

死链问题通常是指在HashMap中的链表或树形结构中的某个元素失去了对其他元素的正确引用,导致链表或树的结构破裂。这通常在高并发场景下出现,因为多个线程同时对HashMap进行操作时可能会导致数据结构不一致。让我们通过一个简单的例子来说明死链问题。

假设我们有一个HashMap,其初始容量为16,哈希函数为hash(key) % 16。假设现在有两个线程A和B,它们同时对HashMap执行以下操作:

在线程A和线程B并发操作时,假设以下操作顺序发生:

线程A插入(key1, value1),将其放入bucket 1。

线程A开始插入(key2, value2),检查bucket 1并发现已存在一个元素,因此需要将(key2, value2)放在bucket 1的链表头部。此时,线程A已经获取了链表头部的引用。

在线程A完成插入操作之前,线程B开始插入(key3, value3)。线程B检查bucket 1并发现已存在一个元素,因此需要将(key3, value3)放在bucket 1的链表头部。线程B完成了插入操作。

线程A继续完成插入(key2, value2),但此时链表的结构已被线程B修改。由于线程A之前已经获取了链表头部的引用,它会将(key2, value2)插入到原始链表头部的下一个位置。这样,链表中的元素顺序将变为:(key3, value3) -> (key2, value2) -> (key1, value1)。

在这个例子中,虽然(key2, value2)没有丢失,但由于线程A和线程B的并发操作,链表的最终顺序可能与预期不符。在某些情况下,这可能会导致性能下降,例如在查找元素时需要遍历更长的链表。要避免这种问题,可以使用线程安全的数据结构,如ConcurrentHashMap。

https://javaguide.cn/java/collection/java-collection-questions-02.html#hashmap-多线程操作导致死循环问题看这个

HashMap 在高并发场景下可能出现的性能问题以及如何规避这些问题的更多相关文章

  1. Qunar机票技术部就有一个全年很关键的一个指标:搜索缓存命中率,当时已经做到了>99.7%。再往后,每提高0.1%,优化难度成指数级增长了。哪怕是千分之一,也直接影响用户体验,影响每天上万张机票的销售额。 在高并发场景下,提供了保证线程安全的对象、方法。比如经典的ConcurrentHashMap,它比起HashMap,有更小粒度的锁,并发读写性能更好。线程安全的StringBuilder取代S

    Qunar机票技术部就有一个全年很关键的一个指标:搜索缓存命中率,当时已经做到了>99.7%.再往后,每提高0.1%,优化难度成指数级增长了.哪怕是千分之一,也直接影响用户体验,影响每天上万张机 ...

  2. 高并发场景下System.currentTimeMillis()的性能问题的优化 以及SnowFlakeIdWorker高性能ID生成器

    package xxx; import java.sql.Timestamp; import java.util.concurrent.*; import java.util.concurrent.a ...

  3. 高并发场景下System.currentTimeMillis()的性能问题的优化

    高并发场景下System.currentTimeMillis()的性能问题的优化 package cn.ucaner.alpaca.common.util.key; import java.sql.T ...

  4. HttpClient在高并发场景下的优化实战

    在项目中使用HttpClient可能是很普遍,尤其在当下微服务大火形势下,如果服务之间是http调用就少不了跟http客户端找交道.由于项目用户规模不同以及应用场景不同,很多时候可能不需要特别处理也. ...

  5. C++高并发场景下读多写少的解决方案

    C++高并发场景下读多写少的解决方案 概述 一谈到高并发的解决方案,往往能想到模块水平拆分.数据库读写分离.分库分表,加缓存.加mq等,这些都是从系统架构上解决.单模块作为系统的组成单元,其性能好坏也 ...

  6. C++高并发场景下读多写少的优化方案

    概述 一谈到高并发的优化方案,往往能想到模块水平拆分.数据库读写分离.分库分表,加缓存.加mq等,这些都是从系统架构上解决.单模块作为系统的组成单元,其性能好坏也能很大的影响整体性能,本文从单模块下读 ...

  7. Linux的虚拟内存管理-如何分配和释放内存,以提高服务器在高并发情况下的性能,从而降低了系统的负载

    Linux的虚拟内存管理有几个关键概念: Linux 虚拟地址空间如何分布?malloc和free是如何分配和释放内存?如何查看堆内内存的碎片情况?既然堆内内存brk和sbrk不能直接释放,为什么不全 ...

  8. 【转】记录PHP、MySQL在高并发场景下产生的一次事故

    看了一篇网友日志,感觉工作中值得借鉴,原文如下: 事故描述 在一次项目中,上线了一新功能之后,陆陆续续的有客服向我们反应,有用户的个别道具数量高达42亿,但是当时一直没有到证据表示这是,确实存在,并且 ...

  9. 高并发场景下System.currentTimeMillis()的性能优化

    一.前言 System.currentTimeMillis()的调用比new一个普通对象要耗时的多(具体耗时高出多少我也不知道,不过听说在100倍左右),然而该方法又是一个常用方法, 有时不得不使用, ...

  10. MySQL在大数据、高并发场景下的SQL语句优化和"最佳实践"

    本文主要针对中小型应用或网站,重点探讨日常程序开发中SQL语句的优化问题,所谓“大数据”.“高并发”仅针对中小型应用而言,专业的数据库运维大神请无视.以下实践为个人在实际开发工作中,针对相对“大数据” ...

随机推荐

  1. IM开发干货分享:浅谈IM系统中离线消息、历史消息的最佳实践

    本文由融云技术团队原创分享,原题"IM 消息数据存储结构设计",内容有修订. 1.引言 在如今的移动互联网时代,IM类产品已是我们生活中不可或缺的组成部分.像微信.钉钉.QQ等是典 ...

  2. 基于.NET8.0实现RabbbitMQ的Publish/Subscribe发布订阅以及死信队列

    [前言] RabbitMQ提供了五种消息模型,分别是简单模型.工作队列模型.发布/订阅模型.路由模型和主题模型.‌‌ 简单模型(Simple)‌:在这种模式下,一个生产者将消息发送到一个队列,只有一个 ...

  3. python基础学习6和7

    模块类与对象 模块 内置模块 time, random, os, json 第三方模块 requests, pandas, numpy,.... 自定义模块 xxx.py 常见的内置模块 hashli ...

  4. Java内存模型深度剖析

    为什么要有内存模型  在介绍Java内存模型之前,先来看一下到底什么是计算机内存模型,然后再来看Java内存模型在计算机内存模型的基础上做了哪些事情.要说计算机的内存模型,就要说一下一段古老的历史,看 ...

  5. biancheng-Spring MVC-HandlerAdapter

    二.HandlerAdapter 根据 Handler 来找到支持它的 HandlerAdapter,通过 HandlerAdapter 执行这个 Handler 得到 ModelAndView 对象 ...

  6. 用DeepSeek写程序之一:编写在linux终端窗口右上角动态显示时间的c++程序

    一. 简单需求 早前有个需求当SSH进入linux时,希望在终端窗口动太显示当前的时间,原来是用脚本解决的 while sleep 1;do tput sc;tput cup 0 $(($(tput ...

  7. Visual Studio各个版本密钥

    1.VS2012 旗舰版:YKCW6-BPFPF-BT8C9-7DCTH-QXGWC 2.VS2013 旗舰版:BWG7X-J98B3-W34RT-33B3R-JVYW9 专业版:XDM3T-W3T3 ...

  8. .NET最佳实践:webapi返回IAsyncEnumerable提升性能

    什么是IAsyncEnumerable IAsyncEnumerable<T> 是 .NET 中用于表示异步数据流的接口. 它允许你逐个异步地获取数据项,而不是将所有数据一次性加载到内存中 ...

  9. nginx 编译安装及配置解析

    一.编译安装 安装插件 安装需要的组件 yum -y install gcc gcc-c++ pcre pcre-devel zlib zlib-devel openssl openssl-devel ...

  10. linux mint安装eclipse

    安装eclipse之前需要先安装配置jdk,参考上面, 一.Eclipse的下载与安装 1.首先,在Eclipse的官网中下载最新版的Luna SR2http://www.eclipse.org/do ...