HashMap的tableSizeFor方法解析

2024-06-01 12:58

本文主要是介绍HashMap的tableSizeFor方法解析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

HashMap的源码有点小复杂,一次性没办法全部弄懂,所以分多次去理解,看懂一点就记录一点,最后再来写个汇总篇,这里先记录其中的tableSizeFor方法。另外jdk版本是1.8.

上代码

    /*** Returns a power of two size for the given target capacity.*/static final int tableSizeFor(int cap) {int n = cap - 1;n |= n >>> 1;n |= n >>> 2;n |= n >>> 4;n |= n >>> 8;n |= n >>> 16;return (n < 0) ? 1 : (n >= MAXIMUM_CAPACITY) ? MAXIMUM_CAPACITY : n + 1;}

看英文注释大概也知道是什么意思了,返回目标容量对应的2的幂次方大小,举个栗子:
5到8都返回8,9到16都返回16,明白了吧;
那为什么hashMap的数组长度总是要2的幂次方大小呢?
这里简单说下,HashMap在存数据时,是根据key的hash值(实际还做了扰乱处理)与数组长度-1(2的n次幂-1后二进制低位全是1)进行&运算确定数组的位置,全是1能够保证数据均匀分布,如果有一些位置为0,对应的一些数组位置永远都不会有数据,文字面描述的比较抽象,不明白的可以自己动手试一试,这是另外一个话题,这里不多说。

代码解析

找一个数据来走一遍这段代码,例如2^29 + 1,对应的二进制是:00100000000000000000000000000001
第一步:int n = cap - 1;
n:00100000000000000000000000000000
第二步:n |= n >>> 1;

00100000000000000000000000000000
|
00010000000000000000000000000000
=
00110000000000000000000000000000

第三步:n |= n >>> 2;

00110000000000000000000000000000
|
00001100000000000000000000000000
=
00111100000000000000000000000000

第四步:n |= n >>> 4;

00111100000000000000000000000000
|
00000011110000000000000000000000
=
00111111110000000000000000000000

第五步:n |= n >>> 8;

00111111110000000000000000000000
|
00000000001111111100000000000000
=
00111111111111111100000000000000

第六步:n |= n >>> 16;

00111111111111111100000000000000
|
00000000000000000011111111111111
=
00111111111111111111111111111111

第六步:return (n < 0) ? 1 : (n >= MAXIMUM_CAPACITY) ? MAXIMUM_CAPACITY : n + 1;

00111111111111111111111111111111
+
00000000000000000000000000000001
=
01000000000000000000000000000000

最终2^29 + 1得到的结果为2^30,符合我们最初的判断。观察数据我们发现其实这一段代码的逻辑分三个部分,先减一,再将得到的数的二进制数从1开始全部变成1,最后再加一,最终得到的数总是2的n次幂;你可以再拿个小数再走一遍流程,会让你更加清晰,我这里就不再做了。

这篇关于HashMap的tableSizeFor方法解析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1021073

相关文章

Java中Arrays类和Collections类常用方法示例详解

《Java中Arrays类和Collections类常用方法示例详解》本文总结了Java中Arrays和Collections类的常用方法,涵盖数组填充、排序、搜索、复制、列表转换等操作,帮助开发者高... 目录Arrays.fill()相关用法Arrays.toString()Arrays.sort()A

Nginx安全防护的多种方法

《Nginx安全防护的多种方法》在生产环境中,需要隐藏Nginx的版本号,以避免泄漏Nginx的版本,使攻击者不能针对特定版本进行攻击,下面就来介绍一下Nginx安全防护的方法,感兴趣的可以了解一下... 目录核心安全配置1.编译安装 Nginx2.隐藏版本号3.限制危险请求方法4.请求限制(CC攻击防御)

python生成随机唯一id的几种实现方法

《python生成随机唯一id的几种实现方法》在Python中生成随机唯一ID有多种方法,根据不同的需求场景可以选择最适合的方案,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习... 目录方法 1:使用 UUID 模块(推荐)方法 2:使用 Secrets 模块(安全敏感场景)方法

MyBatis-Plus通用中等、大量数据分批查询和处理方法

《MyBatis-Plus通用中等、大量数据分批查询和处理方法》文章介绍MyBatis-Plus分页查询处理,通过函数式接口与Lambda表达式实现通用逻辑,方法抽象但功能强大,建议扩展分批处理及流式... 目录函数式接口获取分页数据接口数据处理接口通用逻辑工具类使用方法简单查询自定义查询方法总结函数式接口

MySQL深分页进行性能优化的常见方法

《MySQL深分页进行性能优化的常见方法》在Web应用中,分页查询是数据库操作中的常见需求,然而,在面对大型数据集时,深分页(deeppagination)却成为了性能优化的一个挑战,在本文中,我们将... 目录引言:深分页,真的只是“翻页慢”那么简单吗?一、背景介绍二、深分页的性能问题三、业务场景分析四、

JAVA中安装多个JDK的方法

《JAVA中安装多个JDK的方法》文章介绍了在Windows系统上安装多个JDK版本的方法,包括下载、安装路径修改、环境变量配置(JAVA_HOME和Path),并说明如何通过调整JAVA_HOME在... 首先去oracle官网下载好两个版本不同的jdk(需要登录Oracle账号,没有可以免费注册)下载完

nginx -t、nginx -s stop 和 nginx -s reload 命令的详细解析(结合应用场景)

《nginx-t、nginx-sstop和nginx-sreload命令的详细解析(结合应用场景)》本文解析Nginx的-t、-sstop、-sreload命令,分别用于配置语法检... 以下是关于 nginx -t、nginx -s stop 和 nginx -s reload 命令的详细解析,结合实际应

MyBatis中$与#的区别解析

《MyBatis中$与#的区别解析》文章浏览阅读314次,点赞4次,收藏6次。MyBatis使用#{}作为参数占位符时,会创建预处理语句(PreparedStatement),并将参数值作为预处理语句... 目录一、介绍二、sql注入风险实例一、介绍#(井号):MyBATis使用#{}作为参数占位符时,会

Java中读取YAML文件配置信息常见问题及解决方法

《Java中读取YAML文件配置信息常见问题及解决方法》:本文主要介绍Java中读取YAML文件配置信息常见问题及解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要... 目录1 使用Spring Boot的@ConfigurationProperties2. 使用@Valu

Java 方法重载Overload常见误区及注意事项

《Java方法重载Overload常见误区及注意事项》Java方法重载允许同一类中同名方法通过参数类型、数量、顺序差异实现功能扩展,提升代码灵活性,核心条件为参数列表不同,不涉及返回类型、访问修饰符... 目录Java 方法重载(Overload)详解一、方法重载的核心条件二、构成方法重载的具体情况三、不构