从源代码看 HashMap 的加载因子和容量分配

本贴最后更新于 2384 天前,其中的信息可能已经水流花落

一直知道 HashMap 有默认的容量和加载因子,今天想看看源代码,希望能了解的更清楚一些。 



我们先看看默认的构造器吧,以下为我本机的 JDK6.0 的源代码. 


  1.     /**
  2.      * 默认的初始化的容量,必须是2的幂次数<br>
  3.      * The default initial capacity - MUST be a power of two.
  4.      */
  5.     static final int DEFAULT_INITIAL_CAPACITY = 16;

  6.     /**
  7.      * 默认的加载因子
  8.      */
  9.     static final float DEFAULT_LOAD_FACTOR = 0.75f;

  10.     /**
  11.      * 下一个需要重新分配的尺寸值。等于容量乘以加载因子。<br>
  12.      * 也就是说,一旦容量到了这个数值,将重新分配容器的尺寸。
  13.      * The next size value at which to resize (capacity * load factor).
  14.      * @serial
  15.      */
  16.     int threshold;

  17.     public HashMap() {
  18.         this.loadFactor = DEFAULT_LOAD_FACTOR; 
  19.         threshold = (int)(DEFAULT_INITIAL_CAPACITY * DEFAULT_LOAD_FACTOR);
  20.         table = new Entry[DEFAULT_INITIAL_CAPACITY];
  21.         init();
  22.     }


从代码可以看出,默认的容量是16,而 threshold是16*0.75 = 12; 
我们来看看增加的部分代码。 
  1.     public V put(K key, V value) {
  2.         // 我们忽略掉这部分的代码,只看我们这里最关心的部分
  3.         addEntry(hash, key, value, i); // 这里增加了一个Entry,我们看看代码
  4.         return null;
  5.     }

  6.     void addEntry(int hash, K key, V value, int bucketIndex) {
  7.     Entry<K,V> e = table[bucketIndex];
  8.         table[bucketIndex] = new Entry<K,V>(hash, key, value, e);
  9.         if (size++ >= threshold) // 这里是关键,一旦大于等于threshold的数值
  10.             resize(2 * table.length); // 将会引起容量2倍的扩大
  11.     }

  12.     void resize(int newCapacity) {
  13.         Entry[] oldTable = table;
  14.         int oldCapacity = oldTable.length;
  15.         if (oldCapacity == MAXIMUM_CAPACITY) {
  16.             threshold = Integer.MAX_VALUE;
  17.             return;
  18.         }

  19.         Entry[] newTable = new Entry[newCapacity]; // 新的容器空间
  20.         transfer(newTable); // 复制数据过去
  21.         table = newTable;
  22.         threshold = (int)(newCapacity * loadFactor); // 重新计算threshold的值
  23.     }
好了,我想我们已经清楚大部分了。 
其中有一点,起始容量必须是2的幂次,这如何保证呢?我们来看看其构造方法
  1.     public HashMap(int initialCapacity, float loadFactor) {
  2.         // 忽略掉一部分代码....

  3.         // Find a power of 2 >= initialCapacity
  4.         // 重新查找不比指定数值大的最大的2的幂次数
  5.         int capacity = 1;
  6.         while (capacity < initialCapacity)
  7.             capacity <<= 1;
  8.         // 其它的初始化代码 ...
  9.     }
好了,关于起始容量和加载因子的探讨我们就到这里了。我们应该有了一定的了解了。 

总结: 
    相对准确的估算数据量,将极大的影响HashMap的性能,因为resize是一个重新分配的过程,耗时应该是里面最大的。 
    加载因子较小,会有更多的空间空闲,我不知道这个0.75是不是一个折中方案。也许0.9也是一个不错的选择,特别是那些数据量虽然很大,但不是经常变化的地方,比如公司人员,城市列表等相对比较固定的数据

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...
  • someone756 via macOS

    能不能好好排排版?直接粘贴复制真的好么???????????????!!!!!!!!!!!!!!!!!!!

  • someone756 via macOS

    有意义么?????

推荐标签 标签

  • ngrok

    ngrok 是一个反向代理,通过在公共的端点和本地运行的 Web 服务器之间建立一个安全的通道。

    7 引用 • 63 回帖 • 654 关注
  • JavaScript

    JavaScript 一种动态类型、弱类型、基于原型的直译式脚本语言,内置支持类型。它的解释器被称为 JavaScript 引擎,为浏览器的一部分,广泛用于客户端的脚本语言,最早是在 HTML 网页上使用,用来给 HTML 网页增加动态功能。

    730 引用 • 1282 回帖 • 3 关注
  • DNSPod

    DNSPod 建立于 2006 年 3 月份,是一款免费智能 DNS 产品。 DNSPod 可以为同时有电信、网通、教育网服务器的网站提供智能的解析,让电信用户访问电信的服务器,网通的用户访问网通的服务器,教育网的用户访问教育网的服务器,达到互联互通的效果。

    6 引用 • 26 回帖 • 539 关注
  • 叶归
    12 引用 • 56 回帖 • 20 关注
  • 微服务

    微服务架构是一种架构模式,它提倡将单一应用划分成一组小的服务。服务之间互相协调,互相配合,为用户提供最终价值。每个服务运行在独立的进程中。服务于服务之间才用轻量级的通信机制互相沟通。每个服务都围绕着具体业务构建,能够被独立的部署。

    96 引用 • 155 回帖
  • Eclipse

    Eclipse 是一个开放源代码的、基于 Java 的可扩展开发平台。就其本身而言,它只是一个框架和一组服务,用于通过插件组件构建开发环境。

    76 引用 • 258 回帖 • 628 关注
  • 小说

    小说是以刻画人物形象为中心,通过完整的故事情节和环境描写来反映社会生活的文学体裁。

    32 引用 • 108 回帖
  • H2

    H2 是一个开源的嵌入式数据库引擎,采用 Java 语言编写,不受平台的限制,同时 H2 提供了一个十分方便的 web 控制台用于操作和管理数据库内容。H2 还提供兼容模式,可以兼容一些主流的数据库,因此采用 H2 作为开发期的数据库非常方便。

    11 引用 • 54 回帖 • 672 关注
  • 正则表达式

    正则表达式(Regular Expression)使用单个字符串来描述、匹配一系列遵循某个句法规则的字符串。

    31 引用 • 94 回帖 • 1 关注
  • MongoDB

    MongoDB(来自于英文单词“Humongous”,中文含义为“庞大”)是一个基于分布式文件存储的数据库,由 C++ 语言编写。旨在为应用提供可扩展的高性能数据存储解决方案。MongoDB 是一个介于关系数据库和非关系数据库之间的产品,是非关系数据库当中功能最丰富,最像关系数据库的。它支持的数据结构非常松散,是类似 JSON 的 BSON 格式,因此可以存储比较复杂的数据类型。

    91 引用 • 59 回帖 • 4 关注
  • Git

    Git 是 Linux Torvalds 为了帮助管理 Linux 内核开发而开发的一个开放源码的版本控制软件。

    211 引用 • 358 回帖
  • Wide

    Wide 是一款基于 Web 的 Go 语言 IDE。通过浏览器就可以进行 Go 开发,并有代码自动完成、查看表达式、编译反馈、Lint、实时结果输出等功能。

    欢迎访问我们运维的实例: https://wide.b3log.org

    30 引用 • 218 回帖 • 643 关注
  • Sublime

    Sublime Text 是一款可以用来写代码、写文章的文本编辑器。支持代码高亮、自动完成,还支持通过插件进行扩展。

    10 引用 • 5 回帖 • 1 关注
  • 音乐

    你听到信仰的声音了么?

    62 引用 • 512 回帖
  • Ubuntu

    Ubuntu(友帮拓、优般图、乌班图)是一个以桌面应用为主的 Linux 操作系统,其名称来自非洲南部祖鲁语或豪萨语的“ubuntu”一词,意思是“人性”、“我的存在是因为大家的存在”,是非洲传统的一种价值观,类似华人社会的“仁爱”思想。Ubuntu 的目标在于为一般用户提供一个最新的、同时又相当稳定的主要由自由软件构建而成的操作系统。

    127 引用 • 169 回帖 • 1 关注
  • TensorFlow

    TensorFlow 是一个采用数据流图(data flow graphs),用于数值计算的开源软件库。节点(Nodes)在图中表示数学操作,图中的线(edges)则表示在节点间相互联系的多维数据数组,即张量(tensor)。

    20 引用 • 19 回帖
  • CSS

    CSS(Cascading Style Sheet)“层叠样式表”是用于控制网页样式并允许将样式信息与网页内容分离的一种标记性语言。

    198 引用 • 543 回帖 • 1 关注
  • V2EX

    V2EX 是创意工作者们的社区。这里目前汇聚了超过 400,000 名主要来自互联网行业、游戏行业和媒体行业的创意工作者。V2EX 希望能够成为创意工作者们的生活和事业的一部分。

    16 引用 • 236 回帖 • 250 关注
  • Dubbo

    Dubbo 是一个分布式服务框架,致力于提供高性能和透明化的 RPC 远程服务调用方案,是 [阿里巴巴] SOA 服务化治理方案的核心框架,每天为 2,000+ 个服务提供 3,000,000,000+ 次访问量支持,并被广泛应用于阿里巴巴集团的各成员站点。

    60 引用 • 82 回帖 • 614 关注
  • Mac

    Mac 是苹果公司自 1984 年起以“Macintosh”开始开发的个人消费型计算机,如:iMac、Mac mini、Macbook Air、Macbook Pro、Macbook、Mac Pro 等计算机。

    167 引用 • 597 回帖
  • GitLab

    GitLab 是利用 Ruby 一个开源的版本管理系统,实现一个自托管的 Git 项目仓库,可通过 Web 界面操作公开或私有项目。

    46 引用 • 72 回帖
  • uTools

    uTools 是一个极简、插件化、跨平台的现代桌面软件。通过自由选配丰富的插件,打造你得心应手的工具集合。

    7 引用 • 28 回帖 • 2 关注
  • 机器学习

    机器学习(Machine Learning)是一门多领域交叉学科,涉及概率论、统计学、逼近论、凸分析、算法复杂度理论等多门学科。专门研究计算机怎样模拟或实现人类的学习行为,以获取新的知识或技能,重新组织已有的知识结构使之不断改善自身的性能。

    77 引用 • 37 回帖 • 1 关注
  • 爬虫

    网络爬虫(Spider、Crawler),是一种按照一定的规则,自动地抓取万维网信息的程序。

    106 引用 • 275 回帖 • 2 关注
  • Latke

    Latke 是一款以 JSON 为主的 Java Web 框架。

    71 引用 • 535 回帖 • 829 关注
  • CodeMirror
    2 引用 • 17 回帖 • 166 关注
  • RemNote
    2 引用 • 16 回帖 • 21 关注