Java 对象的哈希值是每次 hashCode() 方法调用重计算么?
16lz
2021-02-15
对于没有覆盖hashCode()
方法的对象
如果没有覆盖 hashCode()
方法,那么哈希值为底层 JDK C++ 源码实现,实例每次调用hashcode()
方法,只有第一次计算哈希值,之后哈希值会存储在对象头的 标记字(MarkWord) 中。
如果进入各种锁状态,那么会缓存在其他地方,一般是获取锁的线程里面存储,恢复无锁(即释放锁)会改回原有的哈希值。
对应源码synchronizer.cpp
:
//如果是无锁状态if (mark.is_neutral()) { hash = mark.hash(); //如果hash不等于0,证明计算过,直接返回 if (hash != 0) { return hash; } //否则,计算hash值 hash = get_next_hash(self, obj); // get a new hash //拷贝到Header中记录 temp = mark.copy_set_hash(hash); test = obj->cas_set_mark(temp, mark); //可能有并发,而且不同默认哈希值计算方法,可能每次哈希值不一样,只有 CAS 成功的才是最后的哈希值 //默认的哈希值计算,不论计算多少次,都不会变 if (test == mark) { return hash; }} else if (mark.has_monitor()) { //如果是有 monitor 锁状态(重量级锁),则获取其 monitor,哈希值会记录在monitor的头部 monitor = mark.monitor(); temp = monitor->header(); assert(temp.is_neutral(), "invariant: header=" INTPTR_FORMAT, temp.value()); hash = temp.hash(); if (hash != 0) { OrderAccess::loadload_for_IRIW(); if (monitor->is_being_async_deflated()) { monitor->install_displaced_markword_in_object(obj); continue; } return hash; }} else if (self->is_lock_owned((address)mark.locker())) { // 如果是轻量级锁状态,获取轻量锁,其中也记录着之前计算的哈希值 temp = mark.displaced_mark_helper(); assert(temp.is_neutral(), "invariant: header=" INTPTR_FORMAT, temp.value()); hash = temp.hash(); if (hash != 0) { // if it has a hash, just return it return hash; }}
对于已经覆盖hashCode()
方法的对象
对于已经覆盖hashCode()
方法的对象,则每次都会重新调用hashCode()
方法重新计算哈希值。
©著作权归作者所有:来自51CTO博客作者zhxdick的原创作品,谢绝转载,否则将追究法律责任微信搜索“我的编程喵”关注公众号,每日一刷,轻松提升技术,斩获各种offer:
好知识,才能预见未来
赞赏
0人进行了赞赏支持
更多相关文章
- 计算变量长度
- 这样规范写代码,同事直呼“666”
- JUC的世界II
- Java程序员必备基础:Object的十二个知识点
- 在 vue 中使用条件渲染语句实现 todolist,以及 vue 中的计算属性
- 实例演示抽象类和接口的区别
- 定义一个接口和抽象类、对比接口与抽象类的区别与联系
- jQuery 中常用的 DOM 操作以及使用 jQuery 完成跨域请求操作
- 初识 jQuery 与$()工厂函数的四种用法,四种好用的 getter/setter