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

17 阅读3分钟

下面系统性地总结 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 <<fork>>
    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)来优化高并发下的锁性能。