嘿,大家好,我是小米,一个爱研究技术、也爱讲故事的 29 岁大哥哥。
上周我接了一个社招面试,面试官直接抛过来一个问题:“说说 Java 里的 HashMap 和 ConcurrentHashMap 有什么区别?”我心里一紧,心想这不是基础题嘛,结果我开口解释了半天,面试官的眼神却越来越微妙……嗯,这次面试翻车了。
所以,今天我决定好好梳理一下 HashMap 和 ConcurrentHashMap 的区别,希望我的教训能帮到大家!
先想象一个场景:你和朋友们在餐厅点菜,每个人都可以随时往菜单里加菜。
问题来了:如果两个人同时修改菜单,服务员可能会拿到一份有问题的订单,比如一道菜被重复记录,或者有的菜根本没加上。
在单线程中,HashMap 这个"菜单"工作得很好。但到了多线程环境中,问题就来了:它本身不是线程安全的,多个线程同时操作会导致数据不一致。
于是,Java 提供了一个改进版的“菜单”——ConcurrentHashMap。不仅线程安全,还能保持一定的性能。
HashMap 的底层是由数组和链表组成的,Java 8 以后为了提升性能,又在链表长度超过一定阈值时将链表转换为红黑树。
它的默认容量是 16,每次扩容时会翻倍到 32、64……以此类推。
ConcurrentHashMap 的设计比 HashMap 复杂得多。Java 7 时,它使用了 Segment 作为分段锁的机制。Java 8 之后,Segment 被淘汰,改用了一种基于 CAS(Compare-And-Swap)操作和 Synchronized 锁的设计。
简单来说,ConcurrentHashMap 的核心在于 分段和细粒度锁,它的每个桶(bucket)可以独立加锁,从而提高并发性能。
HashMap 没有任何锁机制,完全是无锁设计。在多线程情况下,最典型的问题是死循环,例如两个线程同时触发扩容操作,导致循环链表形成,程序直接挂掉。
ConcurrentHashMap 使用了锁分段技术来实现线程安全:
另外,它还有一个巧妙的设计:分段锁(Segmented Lock)。每个桶对应一个锁,多个线程可以同时操作不同的桶,避免了全表加锁的性能损耗。
HashMap 的性能很高,因为它根本没有锁,单线程环境下表现优秀。但在多线程环境中会产生数据不一致的问题。
ConcurrentHashMap 引入了锁机制,多线程安全性大大提高,但性能上会稍逊于 HashMap。不过,Java 8 的优化让它在高并发环境中表现得非常高效。
其实从使用层面来看,HashMap 和 ConcurrentHashMap 的 API 非常相似。
HashMap 的常用方法:
ConcurrentHashMap 的特有方法:
这些增强版方法尤其适合在并发场景中使用。
“HashMap 和 ConcurrentHashMap 的区别可以从以下几个方面来看:”
1、线程安全性:
2、底层结构:
3、并发性能:
4、API 支持:
结尾感悟:面试翻车也是成长的机会
虽然那次面试没通过,但这个问题让我有了更深的认识,也提醒我:技术上的基础知识不能掉以轻心。
原网址: 访问
创建于: 2025-01-14 11:22:40
目录: default
标签: 无
未标明原创文章均为采集,版权归作者所有,转载无需和我联系,请注明原出处,南摩阿彌陀佛,知识,不只知道,要得到
java windows火焰图_mob64ca12ec8020的技术博客_51CTO博客 - 在windows下不可行,不知道作者是怎样搞的 监听SpringBoot 服务启动成功事件并打印信息_监听springboot启动完毕-CSDN博客 SpringBoot中就绪探针和存活探针_management.endpoint.health.probes.enabled-CSDN博客 u2u转换板 - 嘉立创EDA开源硬件平台 Spring Boot 项目的轻量级 HTTP 客户端 retrofit 框架,快来试试它!_Java精选-CSDN博客 手把手教你打造一套最牛的知识笔记管理系统! - 知乎 - 想法有重合-理论可参考 安宇雨 闲鱼 机械键盘 客制化 开贴记录 文本 linux 使用find命令查找包含某字符串的文件_beijihukk的博客-CSDN博客_find 查找字符串 ---- mac 也适用 安宇雨 打字音 记录集合 B站 bilibili 自行搭建 开坑 真正的客制化 安宇雨 黑苹果开坑 查找工具包maven pom 引用地 工具网站 Dantelis 介绍的玩轴入坑攻略 --- 关于轴的一些说法 --- 非官方 ---- 心得而已 --- 长期开坑更新 [本人问题][新开坑位]关于自动化测试的工具与平台应用 机械键盘 开团 网站记录 -- 能做一个收集的程序就好了 不过现在没时间 -- 信息大多是在群里发的 - 你要让垃圾佬 都去一个地方看难度也是很大的 精神支柱 [超级前台]sprinbboot maven superdesk-app 记录 [信息有用] [环境准备] [基本完成] [sebp/elk] 给已创建的Docker容器增加新的端口映射 - qq_30599553的博客 - CSDN博客 [正在研究] Elasticsearch, Logstash, Kibana (ELK) Docker image documentation elasticsearch centos 安装记录 及 启动手记 正式服务器 39 elasticsearch 问题合集 不断更新 6.1.1 | 6.5.1 两个版本 博客程序 - 测试 - bug记录 等等问题 laravel的启动过程解析 - lpfuture - 博客园 OAuth2 Server PHP 用 Laravel 搭建带 OAuth2 验证的 RESTful 服务 | Laravel China 社区 - 高品质的 Laravel 和 PHP 开发者社区 利用Laravel 搭建oauth2 API接口 附 Unauthenticated 解决办法 - 煮茶的博客 - SegmentFault 思否 使用 OAuth2-Server-php 搭建 OAuth2 Server - 午时的海 - 博客园 基于PHP构建OAuth 2.0 服务端 认证平台 - Endv - 博客园 Laravel 的 Artisan 命令行工具 Laravel 的文件系统和云存储功能集成 浅谈Chromium中的设计模式--终--Observer模式 浅谈Chromium中的设计模式--二--pre/post和Delegate模式 浅谈Chromium中的设计模式--一--Chromium中模块分层和进程模型 DeepMind 4 Hacking Yourself README.md update 20211011
Laravel China 简书 知乎 博客园 CSDN博客 开源中国 Go Further Ryan是菜鸟 | LNMP技术栈笔记 云栖社区-阿里云 Netflix技术博客 Techie Delight Linkedin技术博客 Dropbox技术博客 Facebook技术博客 淘宝中间件团队 美团技术博客 360技术博客 古巷博客 - 一个专注于分享的不正常博客 软件测试知识传播 - 测试窝 有赞技术团队 阮一峰 语雀 静觅丨崔庆才的个人博客 软件测试从业者综合能力提升 - isTester IBM Java 开发 使用开放 Java 生态系统开发现代应用程序 pengdai 一个强大的博主 HTML5资源教程 | 分享HTML5开发资源和开发教程 蘑菇博客 - 专注于技术分享的博客平台 个人博客-leapMie 流星007 CSDN博客 - 舍其小伙伴 稀土掘金 Go 技术论坛 | Golang / Go 语言中国知识社区
最新评论