Java对象的哈希值是每次hashCode()方法调用重计算吗?

对于没有覆盖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()方法重新计算哈希值。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值