Java HashMap 在获得 Key 的 Hash 值的时候用的是什么算法
Java 在 HashMap Key 的 Hash 值的时候用的的是自己 Object 中的 hashCode() 算法。
返回的结果是一个整数值。
如果你查看 JDK 的源代码的话,在 HashMap 类中会有下面的这个方法。
public final int hashCode() {
return Objects.hashCode(key) ^ Objects.hashCode(value);
}
通过这个方法,我们可以看到返回的值是整数。
JDK hashCode
如果我们继续跟踪代码,我们会看到在最最基础的 Object 对象中。
有下面的代码:
public int hashCode() {
return J9VMInternals.fastIdentityHashCode(this);
}
是不是很奇怪,为什么这个类的名字为:J9VMInternals
这是因为我们的机器装的是 IBM 的 OpenJ9 虚拟机的版本。
IBM 在这里定义了自己的获得 hashCode 的方法。
static int fastIdentityHashCode(Object anObject) {
com.ibm.jit.JITHelpers h = jitHelpers;
if (null == h) {
return identityHashCode(anObject); /* use early returns to make the JIT code faster */
}
if (h.is32Bit()) {
int ptr = h.getIntFromObject(anObject, 0L);
if ((ptr & com.ibm.oti.vm.VM.OBJECT_HEADER_HAS_BEEN_MOVED_IN_CLASS) != 0) {
if (!h.isArray(anObject)) {
int j9class = ptr & com.ibm.oti.vm.VM.J9_JAVA_CLASS_MASK;
return h.getIntFromObject(anObject, h.getBackfillOffsetFromJ9Class32(j9class));
}
}
} else {
long ptr = (com.ibm.oti.vm.VM.FJ9OBJECT_SIZE == 4) ? Integer.toUnsignedLong(h.getIntFromObject(anObject, 0L)) : h.getLongFromObject(anObject, 0L);
if ((ptr & com.ibm.oti.vm.VM.OBJECT_HEADER_HAS_BEEN_MOVED_IN_CLASS) != 0) {
if (!h.isArray(anObject)) {
long j9class = ptr & com.ibm.oti.vm.VM.J9_JAVA_CLASS_MASK;
return h.getIntFromObject(anObject, h.getBackfillOffsetFromJ9Class64(j9class));
}
}
}
return identityHashCode(anObject);
}
这个 hashCode 的方法被 IBM 提高了下,主要是看看输入的对象是不是 32 位的,如果是 32 位的,会用到 IBM 自己的方法。
否则还是使用传统的 hashCode 方法。
当然,对程序员来说,这部分的内容是透明的,程序员通常只需要知道 JDK 会在你对对象获得 hashCode 的时候返回一个整数值。
在 HashMap 插入数据的时候需要计算 Hash 值,这个方法也会被用到。
可以说这个方法是 JDK 的基础的基础了。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 无需6万激活码!GitHub神秘组织3小时极速复刻Manus,手把手教你使用OpenManus搭建本
· C#/.NET/.NET Core优秀项目和框架2025年2月简报
· Manus爆火,是硬核还是营销?
· 终于写完轮子一部分:tcp代理 了,记录一下
· 【杭电多校比赛记录】2025“钉耙编程”中国大学生算法设计春季联赛(1)
2021-12-01 Git 如何清理敏感文本
2020-12-01 GitHub 如何为项目设置许可证
2020-12-01 GitHub 如何让你的提交显示被校验
2020-12-01 如何在 Windows 上创建一个新的 GPG key
2018-12-01 Confluence 6 空间标识
2018-12-01 Confluence 6 从一个模板中创建一个空间
2018-12-01 从早期 Spring Boot 版本升级