Java 线程状态变化与ObjectMonitor之间的关系

下面系统性地总结 Java 线程状态变化_owner_cxq_EntryList_WaitSet 四个重量级锁核心结构之间的交互。


一、线程状态与四个结构的关系概览

线程状态 所属结构 说明
RUNNABLE (持有锁) _owner 当前持有锁的线程,正在执行同步块
BLOCKED _cxq_EntryList 等待获取锁的线程,处于阻塞状态
WAITING / TIMED_WAITING _WaitSet 已持有锁但主动调用 wait(),等待被唤醒
NEW / TERMINATED 尚未参与锁竞争,或已结束

二、状态转换与结构交互的详细流程

1️⃣ NEW → RUNNABLE(未涉及锁)

  • 不涉及 _owner / _cxq / _EntryList / _WaitSet

2️⃣ 线程尝试进入 synchronized 并成功获得锁

  • 前提_owner == NULL
  • 操作 :通过 CAS 将 _owner 设置为当前线程。
  • 状态 :线程变为 RUNNABLE(持有锁)。
  • 结构变化_owner = currentThread

3️⃣ 线程尝试进入 synchronized 但锁已被占用 → RUNNABLE → BLOCKED

  • 前提_owner != NULL 且不是当前线程。
  • 操作
    1. 当前线程被封装为 ObjectWaiter 节点。
    2. 通过 CAS 头插 将节点加入 _cxq(LIFO 栈)。
    3. 可能先自旋,若仍失败则调用 park() 阻塞。
  • 状态 :从 RUNNABLE 变为 BLOCKED
  • 结构变化 :线程加入 _cxq 头部,_owner 不变。

4️⃣ 锁释放(_owner 退出 synchronized)→ 唤醒 _EntryList_cxq 中的线程

  • 前提 :当前线程是 _owner,即将释放锁。
  • 操作 (以默认 QMode=0 为例):
    1. 如果 _EntryList 非空 → 取出头部线程唤醒。
    2. 如果 _EntryList 为空 → _cxq 中所有线程批量转移到 _EntryList (通常反转顺序,使 LIFO 变 FIFO),然后唤醒 _EntryList 头部线程。
  • 被唤醒的线程 :从 BLOCKED 变为 RUNNABLE(尝试 CAS 成为新 _owner)。
  • 结构变化
    • _owner = NULL
    • _cxq 被清空,_EntryList 获得一批等待线程。

5️⃣ 持有锁的线程调用 Object.wait()RUNNABLE → WAITING

  • 前提 :当前线程是 _owner
  • 操作
    1. 将当前线程加入 _WaitSet 尾部。
    2. 释放锁:设置 _owner = NULL
    3. 调用 park() 阻塞当前线程。
  • 状态 :从 RUNNABLE 变为 WAITING
  • 结构变化
    • _owner = NULL
    • _WaitSet 增加当前线程。
    • 锁释放后,会触发 第 4 步 的唤醒流程。

6️⃣ 其他线程调用 notify() / notifyAll() → 唤醒 _WaitSet 中的线程

  • 前提 :调用 notify() 的线程必须持有同一个对象的锁(即 _owner == 该线程)。
  • 操作 (以 notify() 为例):
    1. _WaitSet 中取出一个线程(通常是头部)。
    2. 将该线程移入 _EntryList (默认策略)或 _cxq(取决于 JVM 参数)。
  • 被移出的线程 :状态仍为 WAITING,直到被唤醒并重新竞争锁。当它被移入 _EntryList 后,其状态实际上变为 BLOCKED(等待锁)。
  • 结构变化
    • _WaitSet 移除该线程。
    • _EntryList(或 _cxq)增加该线程。

7️⃣ 被 notify() 唤醒的线程重新获得锁 → 从 BLOCKED 回到 RUNNABLE

  • 前提 :线程已在 _EntryList(或 _cxq)中等待锁。
  • 操作 :当锁再次被释放且该线程被选为唤醒对象时,它会尝试 CAS 成为 _owner
  • 状态变化BLOCKEDRUNNABLE(持有锁)。
  • 结构变化 :线程从 _EntryList(或 _cxq)中移除,_owner 指向该线程。

8️⃣ 线程终止 → TERMINATED

  • 前提run() 方法结束或异常退出。
  • 操作 :如果该线程是 _owner,则释放锁;如果它在任何队列中,则从队列中移除。
  • 结构变化:清理相关引用。

三、交互关系图(Mermaid)

stateDiagram-v2 [*] --> RUNNABLE_NoLock : start() state RUNNABLE_NoLock <> RUNNABLE_NoLock --> RUNNABLE_Locked : 获得锁\n_owner = current RUNNABLE_Locked --> WAITING : wait()\n加入 _WaitSet\n释放锁\n_owner = NULL WAITING --> BLOCKED_EntryList : notify()\n移到 _EntryList WAITING --> BLOCKED_cxq : notify() (特定策略)\n移到 _cxq RUNNABLE_Locked --> TERMINATED : 结束 RUNNABLE_NoLock --> BLOCKED_cxq : 锁被占\nCAS 入 _cxq BLOCKED_cxq --> BLOCKED_EntryList : 锁释放时\n批量转移 _cxq → _EntryList BLOCKED_EntryList --> RUNNABLE_Locked : 被唤醒并获得锁\n从 _EntryList 移除\n_owner = current

四、核心原则总结

状态转换 结构变化核心点
获得锁 _owner = current
释放锁 _owner = NULL,并触发 _cxq → _EntryList 转移
锁竞争失败 线程加入 _cxq 头部
调用 wait() _owner → 加入 _WaitSet_owner 置空
调用 notify() _WaitSet → 移到 _EntryList(或 _cxq
批量转移 锁释放时,若 _EntryList 空,则 _cxq 全体移至 _EntryList(通常反转顺序)

重要提醒_cxq_EntryList 中的线程状态均为 BLOCKED_WaitSet 中的线程状态为 WAITING / TIMED_WAITING_owner 始终指向 RUNNABLE 且持有锁的线程(或 NULL)。

这个交互模型解释了 HotSpot JVM 如何通过分离新竞争者(_cxq)和锁候选者(_EntryList)来优化高并发下的锁性能。

相关推荐
用户6073203694510 小时前
Python 入门必备-pip install 常用命令例子大全:从基础安装到国内镜像加速实战
后端
小小小前端啊10 小时前
前端网络知识指南
后端
2301_7796224110 小时前
SQL分组聚合优化_GROUP BY索引与优化方案
jvm·数据库·python
m0_7407963610 小时前
golang如何使用sync.WaitGroup_golang sync.WaitGroup并发等待使用方法
jvm·数据库·python
2401_8242226910 小时前
c++如何通过重定向rdbuf来捕获第三方库的日志输出到文件【详解】
jvm·数据库·python
2401_8676239810 小时前
CSS如何解决响应式文字大小调整_利用clamp函数实现流体排版
jvm·数据库·python
2501_9010064710 小时前
如何使用SQL视图快速生成测试数据_模拟复杂场景
jvm·数据库·python
2401_8504916510 小时前
安装宝塔面板提示端口被占用_查找并终止占用进程
jvm·数据库·python
2401_8330336210 小时前
如何通过Java存储过程执行OS命令_Runtime.getRuntime().exec的封装与安全限制
jvm·数据库·python
weixin_4597539410 小时前
SymPy中正确处理含整数参数的三角函数定积分:避免n=0特例干扰结果
jvm·数据库·python