0

0

如何在Java中理解对象的锁状态_对象头Mark Word中的锁标志位

P粉602998670

P粉602998670

发布时间:2026-03-16 13:34:31

|

511人浏览过

|

来源于php中文网

原创

Mark Word的锁标志位位于对象头前8字节(64位JVM)中一个4-bit字段,与GC年龄、哈希码、偏向线程ID共用同一内存区域,通过位运算动态复用,其值需用jol工具或JVM诊断参数实时观测。

如何在java中理解对象的锁状态_对象头mark word中的锁标志位

Java对象头里Mark Word的锁标志位到底存哪儿

它不在你写的任何Java代码里,也不在Class文件结构中显式定义——而是JVM在堆上为每个对象分配内存时,悄悄塞进对象头前8字节(64位虚拟机)里的一个4-bit字段。这个字段叫lock bits,和GC分代年龄、哈希码、偏向线程ID挤在同一块Mark Word里,靠位运算动态复用。

常见错误现象:用Unsafe.objectFieldOffset()去查markWord偏移量失败,或者反编译class看到hashCode()返回值和锁状态“对不上”,其实是同一片内存被不同阶段覆盖了。

  • Mark Word不是固定结构,开启-XX:+UseBiasedLocking时前54位可能存偏向线程ID,关了就腾出来放轻量级锁指针
  • 调用Object.hashCode()会强制将哈希码写入Mark Word,如果此时对象已加锁,就会触发锁膨胀并把哈希码挪到对象体外(比如java.lang.Class_hash字段)
  • 32位JVM下Mark Word只有4字节,锁标志位更紧凑,但兼容性差,现在基本只在嵌入式场景见得到

怎么验证当前对象处于哪种锁状态

不能靠synchronized语法糖猜,得看JVM运行时实际布局。最直接的方式是用jol(Java Object Layout)工具打印内存结构:

mvn org.openjdk.jol:jol-cli:0.16:run -Dexec.args="com.example.MyObj"

输出里找mark:那一行,末尾几个bit就是锁标志位。例如:0x0000000000000001表示无锁态,0x0000000000000005末两位01代表轻量级锁(具体编码见HotSpot源码markOop.hpp里的lock_mask_in_place)。

立即学习Java免费学习笔记(深入)”;

OpenJobs AI
OpenJobs AI

AI驱动的职位搜索推荐平台

下载
  • 必须用-XX:+UnlockDiagnosticVMOptions -XX:+PrintAssembly才能看到锁升级时Mark Word的实时变化,普通日志不输出这个细节
  • VisualVMJMC只能看到线程阻塞统计,看不到单个对象的Mark Word
  • 自己写Unsafe读取Mark Word要绕过JVM安全限制,且不同JDK版本字段偏移量可能变动,不推荐线上用

偏向锁关闭后,轻量级锁和重量级锁怎么切换

切换不是由Java代码控制的,而是JVM在同步块入口处通过CAS操作Mark Word失败次数决定的。关键阈值在BiasedLockingStartupDelayBiasedLockingBulkRebiasThreshold这些参数里,但真正起作用的是ObjectSynchronizer::inflate()函数内部逻辑。

  • 轻量级锁CAS失败4次(默认值,由SpinLimit控制)后,JVM认为竞争变高,开始自旋;自旋再失败就膨胀成重量级锁
  • 重量级锁一旦建立,Mark Word里存的是指向ObjectMonitor结构的指针,而ObjectMonitor又包含_owner_WaitSet等字段,这部分内存不在对象头里
  • 调用wait()notify()会强制触发锁膨胀,哪怕之前一直是无锁或偏向态

为什么调试时看到的锁状态和预期不一致

因为Mark Word内容随时在变:GC移动对象会重写Mark Word,线程调度可能导致锁释放瞬间被其他线程抢占,甚至JIT编译器做了锁消除(-XX:+EliminateLocks),让synchronized根本没生成锁逻辑。

  • jstack看到locked <0x...>只说明线程持有monitor,不反映Mark Word当前值
  • 在G1 GC下,如果对象在年轻代Survivor区反复复制,Mark Word会被重置为初始无锁态,但锁记录可能还留在栈帧里
  • 所有基于Mark Word的分析都依赖“恰好抓到那个时刻”的快照,想稳定复现某一种锁状态,得配合-XX:BiasedLockingStartupDelay=0和手动触发Thread.yield()

真正难的不是看懂那几个bit,而是理解它们背后没有独立生命周期——它只是JVM在特定时机对同一块内存的不同解释方式。改一个参数、换一次GC、甚至多跑几毫秒,Mark Word就可能已经换了四轮含义。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

448

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

606

2023.08.10

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

448

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

606

2023.08.10

class在c语言中的意思
class在c语言中的意思

在C语言中,"class" 是一个关键字,用于定义一个类。想了解更多class的相关内容,可以阅读本专题下面的文章。

931

2024.01.03

python中class的含义
python中class的含义

本专题整合了python中class的相关内容,阅读专题下面的文章了解更多详细内容。

32

2025.12.06

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

786

2023.08.10

Java 并发编程高级实践
Java 并发编程高级实践

本专题深入讲解 Java 在高并发开发中的核心技术,涵盖线程模型、Thread 与 Runnable、Lock 与 synchronized、原子类、并发容器、线程池(Executor 框架)、阻塞队列、并发工具类(CountDownLatch、Semaphore)、以及高并发系统设计中的关键策略。通过实战案例帮助学习者全面掌握构建高性能并发应用的工程能力。

102

2025.12.01

chatgpt使用指南
chatgpt使用指南

本专题整合了chatgpt使用教程、新手使用说明等等相关内容,阅读专题下面的文章了解更多详细内容。

0

2026.03.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.5万人学习

C# 教程
C# 教程

共94课时 | 11.5万人学习

Java 教程
Java 教程

共578课时 | 83.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号