美文网首页
2021-2-15:Java 对象的哈希值是每次 hashCod

2021-2-15:Java 对象的哈希值是每次 hashCod

作者: 干货满满张哈希 | 来源:发表于2021-02-15 08:15 被阅读0次

    对于没有覆盖hashCode()方法的对象

    如果没有覆盖 hashCode() 方法,那么哈希值为底层 JDK C++ 源码实现,实例每次调用hashcode()方法,只有第一次计算哈希值,之后哈希值会存储在对象头的 标记字(MarkWord) 中。

    image

    如果进入各种锁状态,那么会缓存在其他地方,一般是获取锁的线程里面存储,恢复无锁(即释放锁)会改回原有的哈希值

    对应源码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()方法重新计算哈希值。

    微信搜索“我的编程喵”关注公众号,每日一刷,轻松提升技术,斩获各种offer

    相关文章

      网友评论

          本文标题:2021-2-15:Java 对象的哈希值是每次 hashCod

          本文链接:https://www.haomeiwen.com/subject/ufkpxltx.html