HashMap的tableSizeFor解析
我们都知道,对于HashMap来说,数组的容量为2的倍数,但是我们可以在创建map的时候传入一个数组的大小
此时,这个初始化数组大小会传给一个双参的构造器
1. 创建HashMap
public static void main(String[] args) {
HashMap<Integer, String> map = new HashMap<>(10);
}
2. 构造器
public HashMap(int initialCapacity) {
this(initialCapacity, DEFAULT_LOAD_FACTOR);
}
public HashMap(int initialCapacity, float loadFactor) {
if (initialCapacity < 0)
throw new IllegalArgumentException("Illegal initial capacity: " +
initialCapacity);
if (initialCapacity > MAXIMUM_CAPACITY)
initialCapacity = MAXIMUM_CAPACITY;
if (loadFactor <= 0 || Float.isNaN(loadFactor))
throw new IllegalArgumentException("Illegal load factor: " +
loadFactor);
this.loadFactor = loadFactor;
//阈值(HashMap中最多能容纳的元素个数)
// 阈值 = 数组长度 * 负载因子
// 但是注意,刚初始化的时候阈值用来临时保存用户指定的初始容量或者默认初始容量16
// 只有在第一次inflateTable()扩容之后,才会用计算出的正确的容量乘以负载因子赋值
//此时传入的initialCapacity是10,会调用tableSizeFor方法,将其转为大于等于它的一个2的幂次数,例如7转为8,8转为8,10就转为16
this.threshold = tableSizeFor(initialCapacity);
}
3. tableSizeFor具体实现
static final int tableSizeFor(int cap) {
//首先减一操作是为了如果本身就为2的幂次数(8,16)的话,就使其降为一个小于2的幂次的数
//例如我们传入的10,对应的2进制为00000000 00000000 00000000 00001010
int n = cap - 1;
//减后
//00000000 00000000 00000000 00001001
n |= n >>> 1;
//右移后
//00000000 00000000 00000000 00000100
//之后再与自身相或也就是上面两个二进制相或
//00000000 00000000 00000000 00001101
n |= n >>> 2;
//再次右移两位
//00000000 00000000 00000000 00000011
//再或
//00000000 00000000 00000000 00001111
//此时我们可以看到我们想要的效果了
//将低位全部置为1!
//后面这些也是一样的操作,因为int是32位的
n |= n >>> 4;
n |= n >>> 8;
n |= n >>> 16;
//之后对 与完的数字进行+1操作,(这些判断我就不说了)
//00000000 00000000 00000000 00010000
//此时+1后变为了一个2的幂次
return (n < 0) ? 1 : (n >= MAXIMUM_CAPACITY) ? MAXIMUM_CAPACITY : n + 1;
}
4. 接下来我在说两个别的例子
-首先是32,他其实最后的结果要的就是32,对应的二进制是
00000000 00000000 00000000 00100000
-接下来先减一
00000000 00000000 00000000 0001 1111
-此时其实我们不用看了,右移之后或上任何数字,此时最后5位是不会变的,都是5个1
00000000 00000000 00000000 0001 1111
-之后对其进行+1操作,就变回了我们的32
00000000 00000000 00000000 0010 0000
-在接下来是很大的数字,这个数字具体是多少我也没算过。。。也不知道,最大容量MAXIMUM_CAPACITY = 1 << 30,所以应该是没超的
注意观察最高位
0010 0000 0010 0100 0101 1000 0010 0101
-先减一
0010 0000 0010 0100 0101 1000 0010 0100
-在右移1位
0001 0000 0001 0010 0010 1100 0001 0010
-此时上面两个相或
0011 0000 0011 0110 0111 1100 0011 0110
--此时再右移两位
啊!不想右移了,我们看一下结果吧
我们观察一下上面一堆,可以发现我们的结果其实与后面几位无关,只与最高的那个1位有关
-例如我们最近的这个
0010 0000 0010 0100 0101 1000 0010 0101
-我们其实可以省略除了最高的那个1之外的数字,我们将他们全部忽略,使用*代替,表示我们不关心它
001* **** **** **** **** **** **** ****
-之后右移一位
0001 ****
-相或
0011 ****
-之后右移两位
0000 11**
-相或
0011 11**
-右移四位
0000 0011 11** **
-相或
0011 1111 11** **
通过不断的这样的操作,将最高位1之后位全部置为1
之后在+1就可以求出来一个2的幂次了
5. 那有些朋友可能会问了,开始为什么要减一呢?
因为对于以下数字,我们可以看出来,减一实际上并没有什么影响,甚至你减个88都没什么影响,那为什么减呢?
0010 0000 0010 0100 0101 1000 0010 0101
主要是为了解决2的幂次的存在
因为对于2的幂次来说,例如32
00000000 00000000 00000000 00100000
我们如果直接去计算,直接右移取或,右移取或
那么我们得到的结果是
00000000 00000000 00000000 00111111
此时最终+1取2的幂的话结果为64!但是我们要的是32,所以说要先减1,使其成为一个比2的幂次小的数字
00000000 00000000 00000000 01000000
HashMap的tableSizeFor解析的更多相关文章
- JDK8 HashMap 源码解析
HashMap中数据结构 在jdk1.7中,HashMap采用数组+链表(拉链法).因为数组是一组连续的内存空间,易查询,不易增删,而链表是不连续的内存空间,通过节点相互连接,易删除,不易查询.Has ...
- HashMap源码解析 非原创
Stack过时的类,使用Deque重新实现. HashCode和equals的关系 HashCode为hash码,用于散列数组中的存储时HashMap进行散列映射. equals方法适用于比较两个对象 ...
- Java中的容器(集合)之HashMap源码解析
1.HashMap源码解析(JDK8) 基础原理: 对比上一篇<Java中的容器(集合)之ArrayList源码解析>而言,本篇只解析HashMap常用的核心方法的源码. HashMap是 ...
- 最全的HashMap源码解析!
HashMap源码解析 HashMap采用键值对形式的存储结构,每个key对应唯一的value,查询和修改的速度很快,能到到O(1)的平均复杂度.他是非线程安全的,且不能保证元素的存储顺序. 他的关系 ...
- 【转】Java HashMap 源码解析(好文章)
.fluid-width-video-wrapper { width: 100%; position: relative; padding: 0; } .fluid-width-video-wra ...
- 史上最全HashMap红黑树解析
HashMap红黑树解析 红黑树介绍 TreeNode结构 树化的过程 红黑树的左旋和右旋 TreeNode的左旋和右旋 红黑树的插入 TreeNode的插入 红黑树的删除 TreeNode的删除节点 ...
- HashMap源码解析和设计解读
HashMap源码解析 想要理解HashMap底层数据的存储形式,底层原理,最好的形式就是读它的源码,但是说实话,源码的注释说明全是英文,英文不是非常好的朋友读起来真的非常吃力,我基本上看了差不多 ...
- 详解HashMap源码解析(下)
上文详解HashMap源码解析(上)介绍了HashMap整体介绍了一下数据结构,主要属性字段,获取数组的索引下标,以及几个构造方法.本文重点讲解元素的添加.查找.扩容等主要方法. 添加元素 put(K ...
- 【Java深入研究】9、HashMap源码解析(jdk 1.8)
一.HashMap概述 HashMap是常用的Java集合之一,是基于哈希表的Map接口的实现.与HashTable主要区别为不支持同步和允许null作为key和value.由于HashMap不是线程 ...
随机推荐
- springboot分页插件的使用
在springboot工程下的pom.xml中添加依赖 <!--分页 pagehelper --> <dependency> <groupId>com.github ...
- Qt 的MDI 多文档窗口
一.MDI简介 MDI就是多文档界面(Multi-document Interface,MDI)应用程序 MDI就是在主窗口里创建多个同类型的MDI子窗口,这些MDI子窗口在主窗口里显示,并共享主窗口 ...
- C#多线程---线程池的工作者线程
一.线程池简介 创建和销毁线程是一个要耗费大量时间的过程,太多的线程也会浪费内存资源,所以通过Thread类来创建过多的线程反而有损于性能,为了改善这样的问题 ,.net中就引入了线程池. 线程池形象 ...
- vscode如何配置ts的lint,如何配置才能让eslint和prettier不冲突一键格式化代码(vue开发使用)
最近在使用ts,发觉tslint在vscode上使用很不方便,不如eslint一键格式化高效,就想着能不能配置下vscode让其像写js一样爽 这篇文章主要解决2个问题,第一个是如何让vscode使用 ...
- Go版本依赖--版本选择机制
目录 1. 版本选择机制 2.依赖包版本约定 2.1 Go module 之前版本兼容性 2.2 Go module 之后版本兼容性 3. 版本选择机制 3.1 最新版本选择 3.2 最小版本选择 1 ...
- 闭包 panic recover
闭包=函数+外层变量的引用 recover必须搭配defer使用 defer一定要在可能引发panic的语句之前定义
- delta源码阅读
阅读思路: 1.源码编译 2.功能如何使用 3.实现原理 4.源码阅读(通读+记录+分析) 源码结构 源码分析 元数据 位置:org.apache.spark.sql.delta.actions下的a ...
- Kickstart部署之HTTP架构
原文转自:https://www.cnblogs.com/itzgr/p/10029527.html作者:木二 目录 一 准备 1.1 完整架构:Kickstart+DHCP+HTTP+TFTP+PX ...
- 使用 IDEA 配合 Dockerfile 部署 SpringBoot 工程
准备 SpringBoot 工程 新建 SpringBoot 项目,默认的端口是 8080 ,新建 Controller 和 Mapping @RestController public class ...
- GoLang设计模式02 - 工厂模式
工厂模式是一种创建型模式,也是最常用的设计模式之一.调用方通过工厂产出并获取对象,可以不必关注对象创建的细节和构建逻辑. 在工厂模式下,调用方只和工厂进行交互,并告诉工厂具体获取哪种类型的对象.工厂负 ...