Java高频面试之集合-17

hello啊,各位观众姥爷们!!!本baby今天来报道了!哈哈哈哈哈嗝🐶

面试官:JDK 8 对 HashMap 主要做了哪些优化呢?为什么要这么做?


JDK 8 对 HashMap 的主要优化及原因

JDK 8 对 HashMap 的实现进行了多项关键优化,显著提升了其在高冲突场景下的性能内存效率。以下是主要优化点及其设计动机:


一、链表转红黑树(Treeify)

优化内容
当单个桶(Bucket)中的链表长度超过阈值(默认 8)且哈希表容量 ≥ 64 时,链表会被转换为红黑树;当树节点数 ≤ 6 时,红黑树退化为链表。

原因

  • 解决链表过长导致的性能问题
    链表查询的时间复杂度为 O(n),而红黑树的查询复杂度为 O(log n)。在高冲突场景下,树化能显著减少查找时间。
  • 平衡内存与性能
    红黑树节点(TreeNode)的内存开销高于链表节点(Node),因此设置退化的阈值(6)以避免小规模数据下的内存浪费。

源码示例

// 链表转红黑树的条件(容量 ≥ 64 且链表长度 ≥ 8)
if (binCount >= TREEIFY_THRESHOLD - 1) {treeifyBin(tab, hash);break;
}

二、哈希函数优化

优化内容
JDK 8 改进了哈希值计算方式,通过 高位异或(XOR) 增强散列性:

static final int hash(Object key) {int h;return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16);
}

原因

  • 减少哈希冲突
    将哈希码的高 16 位与低 16 位异或,使得更多位数参与索引计算((n - 1) & hash),避免仅依赖低位导致的冲突。
  • 提升分布均匀性
    例如,若容量为 16(二进制 10000),原哈希码低位重复性高,异或高位后分布更均匀。

三、扩容机制优化

优化内容
扩容时,通过 高位掩码判断 元素的新位置,避免重新计算哈希值:

if ((e.hash & oldCap) == 0) {// 新索引 = 原索引
} else {// 新索引 = 原索引 + 原容量
}

原因

  • 减少计算开销
    原扩容需重新计算所有元素的哈希值和索引,JDK 8 直接通过哈希值的特定位判断位置,性能提升显著。
  • 元素均匀拆分
    扩容后,原桶中的元素被均分到两个新桶中(低位桶和高位桶),减少链表或树的深度。

四、树化条件优化

优化内容
链表转红黑树需满足 容量 ≥ 64,否则优先扩容而非树化。

原因

  • 避免小容量下过早树化
    若容量较小(如 16),扩容可有效减少冲突概率,此时树化反而增加内存开销且收益有限。
  • 优先利用扩容分散冲突
    扩容后哈希分布更均匀,可能自然解决冲突,减少树化需求。

五、性能对比与设计权衡
场景JDK 7 链表查询JDK 8 红黑树查询优化收益
链表长度 = 8O(8) → 8次遍历O(log 8) → 3次比较性能提升 60%+
链表长度 = 64O(64) → 64次遍历O(log 64) → 6次比较性能提升 90%+

六、总结与适用场景
优化点解决的问题适用场景
链表转红黑树高冲突下链表查询效率低频繁插入、高哈希冲突的键值对场景
哈希函数优化哈希分布不均导致冲突概率高键的 hashCode() 实现质量参差不齐
扩容机制优化扩容时重新哈希的性能瓶颈大规模数据动态扩容场景

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/898896.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

计算机二级:函数基础题

函数基础题 第一题 rinput("请输入半径:") c3.1415926*r*2 print("{:.0f}".format(c))输出: Type Error第二题 a7 b2 print(a%2)输出 1第三题 ab4 def my_ab(ab,xy):abpow(ab,xy)print(ab,end"\n") my_ab(ab,2)prin…

C# 属性(Property)‌详解

在 C# 中,‌属性(Property)‌ 是类或结构体中的成员,用于封装对私有字段(称为 ‌backing field‌)的访问,提供更灵活和安全的数据操作方式。属性通过 get 和 set 访问器控制对数据的读写&#x…

iPhone 16如何翻译文档?文档翻译技巧、软件推荐

在全球化的今天,跨语言交流变得越来越频繁,而文档翻译更是成为许多人日常工作和学习中的重要需求。作为苹果公司最新推出的旗舰机型,iPhone 16凭借其强大的硬件性能和丰富的软件生态,为我们提供了多种便捷的文档翻译方式&#xff…

HRP方法全文总结与模型流程解析

背景与问题 传统二次优化方法(如Markowitz的CLA)存在三大问题: 不稳定性:协方差矩阵的高条件数导致逆矩阵计算误差放大,权重剧烈波动。 集中性:优化结果过度集中于少数资产,易受个体风险冲击。…

解决项目一直在构建中的问题:以 IntelliJ IDEA 为例提高共享堆内存

在使用 IntelliJ IDEA 时,开发者可能会遇到项目长期处于构建状态的问题。这种情况将严重影响开发效率。通常,这种问题的一个常见原因是构建进程所分配的堆内存不足。本文将以 IntelliJ IDEA 为背景,介绍如何通过提高共享堆内存来解决此问题&a…

金橙子删除打标对象

注意在使用金橙子根据对象名称删除对象时要注意,每删除一个对象,所有对象的索引都将改变。 如果你是用for去遍历,再根据索引获取打标对象名称的话就会出现漏的掉的问题。 改进方法 1,将要删除的对象找到后,统一存放在一个集合中。再根据这个要删除的对象集合再一个个去遍…

JVM常见概念之条件移动

问题 当我们有分支频率数据时,有什么有趣的技巧可以做吗?什么是条件移动? 基础知识 如果您需要在来自一个分支的两个结果之间进行选择,那么您可以在 ISA 级别做两件不同的事情。 首先,你可以创建一个分支&#xff…

MANISKILL3:GPU 并行机器人模拟和渲染,用于通用的具身AI

本文介绍了一种名为ManiSkill3的机器人仿真系统,它采用了GPU并行化技术,并针对通用性进行了优化。该系统支持多种视觉输入方式和异构模拟,能够在物理场景中进行高效的仿真和渲染,达到比其他平台更快的速度和更少的GPU内存使用量。…

计算机网络高频(三)UDP基础

计算机网络高频(三)UDP基础 1.UDP的头部格式是什么样的?⭐ UDP 头部具有以下字段: 源端口(Source Port):16 位字段,表示发送方的端口号。目标端口(Destination Port):16 位字段,表示接收方的端口号。长度(Length):16 位字段,表示 UDP 数据报(包括头部和数据部…

微信小程序中使用Less样式方法

在微信小程序中使用Less样式,可以通过以下步骤实现。主要原理是借助Visual Studio Code(VSCode)的插件将Less文件自动编译为小程序支持的.wxss文件,或通过微信开发者工具的扩展功能直接集成Less编译环境。以下是具体方法&#xff…

Leetcode 刷题笔记 图论part05

卡码网 107 寻找存在的路径 初识并查集 并查集功能: 寻找根节点,函数: find(int u),也就是判断这个节点的祖先节点是哪个将两个节点接入到同一个集合,函数: join(int u, int v),将两个节点连在同一个根节点上判断两…

SpringBoot星之语明星周边产品销售网站设计与实现

在当今数字化时代,明星周边产品的线上销售已成为一种趋势。幽络源作为一站式综合平台,不仅提供免费源码、网络兼职资源,还分享各类技术教程。本文将详细介绍基于SpringBoot的星之语明星周边产品销售网站的设计与实现,帮助开发者快…

怎样对比找到两个git仓库的差异

怎样对比找到两个git仓库的差异 陈拓 2024/12/24-2024/12/28 1. 概述 要比较两个Git仓库的差异,可以使用git diff命令。你需要先将两个仓库的克隆版本都检出到本地,然后在对应的目录中运行git diff命令。 下面我们以YDLIDAR ROS2驱动程序ydlidar_ros2…

C语言-装饰器模式详解与实践 - LED控制系统

文章目录 C语言装饰器模式详解与实践 - LED控制系统1. 什么是装饰器模式?2. 为什么需要装饰器模式?3. 实际应用场景4. 代码实现4.1 头文件 (led_decorator.h)4.2 实现文件 (led_decorator.c)4.3 使用示例 (main.c) 5. 代码分析5.1 关键设计点5.2 实现特点…

Go常见问题与回答(下)

文章目录 1、通过指针变量 p 访问其成员变量 name,有哪几种方式?2、代码,说出结果3、扩容提,代码,说出结果4、指出下面这段代码的错误之处5、是否通过编译6、关于字符串连接,下面语法正确的是7、关于iota&a…

JVM 核心知识点总结

🧑 博主简介:CSDN博客专家,历代文学网(PC端可以访问:https://literature.sinhy.com/#/literature?__c1000,移动端可微信小程序搜索“历代文学”)总架构师,15年工作经验,…

SQL中体会多对多

我们可以根据学生与课程多对多关系的数据库模型,给出实际的表数据以及对应的查询结果示例,会用到JOINLEFT JOIN两种连接 1. 学生表(students) student_idstudent_name1张三2李四3王五 2. 课程表(courses&#xff09…

ES如果要查10条数据需要从各个分片上各取多少条数据?

目录 ES如果要查10条数据需要从各个分片上各取多少条数据? 简单查询(如 match_all 或 term 查询) 深度分页查询(如 from + size 查询) 聚合查询 什么叫聚合查询? 聚合查询的基本结构 常见的聚合类型 聚合查询的执行过程 聚合查询的示例 聚合查询的应用场景 注意…

人机交互自学引导

第1关:输出“Hello World!” # 在下面一行补充代码,输出“Hello World!” print(Hello World!) 第2关:输出“李白,你好!” # 在下面补充代码,在两行中依次输出“李白,你好!”和“…

CentOS 7 更换 yum 源(阿里云)+ 扩展 epel 源

CentOS 7 更换 yum 源(阿里云) 扩展 epel 源 一、备份现有 yum 源二、下载 yum 源(任选其一即可)三、清理并生成缓存四、安装 EPEL 扩展源(根据需要下载)五、验证是否生效六、一键脚本(阿里云源…