news 2026/5/5 21:20:18

AQS共享锁的传播机制精髓

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
AQS共享锁的传播机制精髓

一、背景:共享模式 vs 独占模式

  • 独占模式(Exclusive):如ReentrantLock,一次只允许一个线程持有锁。
    • 释放时:只需唤醒队列中的下一个等待者(调用unparkSuccessor(head)即可)。
  • 共享模式(Shared):如SemaphoreCountDownLatchReadWriteLock的读锁。
    • 允许多个线程同时持有锁(只要资源足够)。
    • 所以:一旦资源可用,可能要连续唤醒多个等待线程!

这就引出了“传播(propagation)”的概念。


二、核心问题:如何确保“所有能获取资源的线程都被唤醒”?

假设:

  • 当前有 3 个线程在 AQS 队列中等待共享锁(A → B → C)。
  • 此时资源释放了(比如信号量 permit +1)。
  • 线程 A 被唤醒并成功获取资源。
  • 但此时资源仍然充足(比如 permit 还剩 1),那么 B 也应该被唤醒!

但如果只在释放时唤醒一次(像独占锁那样),B 和 C 就可能永远等下去,即使资源已经可用。

👉 因此,共享模式必须支持“级联唤醒”或“传播唤醒”


三、PROPAGATE 和 SIGNAL 的作用

🔹Node.SIGNAL(-1)

  • 含义:当前节点的后继节点需要被唤醒
  • 这是 AQS 中通用的状态,独占和共享都用
  • 当一个节点入队后,会把前驱设为SIGNAL,表示“我等着,你释放时记得叫我”。

🔹Node.PROPAGATE(-3)

  • 仅用于共享模式
  • 含义:即使当前没有明确的“需要唤醒”的信号,也要继续传播释放动作
  • 它是一种“保险机制”,防止在并发竞争下漏掉唤醒。

四、为什么需要 PROPAGATE?—— 并发场景下的“信号丢失”问题

考虑这个竞态条件(race condition)

  1. 线程 T1 调用releaseShared(),准备唤醒后继。
  2. 此时队列 head 是 H,H 的 waitStatus 是 0(因为刚被设置为 head,还没来得及设 SIGNAL)。
  3. T1 检查到ws == 0,于是尝试 CAS 把它设为PROPAGATE(表示:“虽然现在没信号,但我要记录这次释放,以便后续传播”)。
  4. 与此同时,另一个线程 T2 成功获取了共享锁,并调用setHeadAndPropagate,把新节点设为 head。
  5. 如果没有PROPAGATE,T1 可能认为“没人需要唤醒”,直接退出,导致后面的线程无法被唤醒!

PROPAGATE的作用就是:在状态不确定时,留下一个“释放发生过”的标记,确保后续操作能继续传播唤醒。


五、代码逻辑精解

setHeadAndPropagate(node, propagate)

  • propagate > 0:表示tryAcquireShared返回正数,说明还有剩余资源,应该继续唤醒别人。
  • 条件判断很“保守”(conservative):
    if(propagate>0||h==null||h.waitStatus<0||...)
    只要有任何迹象表明可能需要传播,就调用doReleaseShared()
  • 特别注意:h.waitStatus < 0包括SIGNAL(-1)PROPAGATE(-3),都表示“需要关注后继”。

doReleaseShared()

这是一个自旋 + CAS 的传播循环

for(;;){Nodeh=head;if(h!=null&&h!=tail){intws=h.waitStatus;if(ws==SIGNAL){// 正常情况:后继需要唤醒CAS(SIGNAL0);unparkSuccessor(h);}elseif(ws==0){// 没有明确信号,但可能是并发释放!// 设为 PROPAGATE,留下“释放已发生”的标记CAS(0PROPAGATE);}}// 如果 head 没变,说明稳定了,可以退出if(h==head)break;}

💡 关键思想:即使当前看不出需要唤醒谁,也要通过 PROPAGATE 确保“释放事件”不会丢失


六、举个实际例子:Semaphore

Semaphoresem=newSemaphore(1);// 三个线程同时调用 sem.acquire()// 初始 permit = 1,只有第一个能成功,其他两个入队等待。sem.release();// permit 变回 1
  • release()→ 调用releaseShared()doReleaseShared()
  • 唤醒第一个等待线程(A)
  • A 获取 permit 后,发现propagate = 0(因为 permit 用完了),不传播
  • 但如果release()被调用了两次(permit=2),则:
    • 第一次唤醒 A,A 获取后 permit=1 →propagate=1→ 继续传播
    • 触发第二次doReleaseShared(),唤醒 B

如果没有传播机制,B 就卡住了!


七、总结:精髓所在

概念作用
传播(Propagation)共享模式下,一次释放可能需唤醒多个线程
SIGNAL (-1)明确指示“后继需要唤醒”
PROPAGATE (-3)在状态模糊时,防止释放信号丢失的保险机制
自旋 + CAS 循环应对高并发下的状态竞争,确保最终一致性

设计哲学
“宁可多唤醒几次(unnecessary wake-ups),也不能漏掉一次该唤醒的线程。”
—— 这就是 AQS 共享模式的鲁棒性所在。


附加:Node.waitStatus 的几个值

常量含义
0初始状态
-1SIGNAL后继需要被唤醒
-2CONDITION在 Condition 队列中
-3PROPAGATE共享模式下,表示应继续传播释放
1CANCELLED节点已取消

希望这能帮你彻底理解 AQS 共享模式的“传播”机制!

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/3 8:00:16

MicroPE官网集成CUDA工具包支持GLM-4.6V-Flash-WEB推理

MicroPE官网集成CUDA工具包支持GLM-4.6V-Flash-WEB推理 在今天这个图像内容爆炸式增长的时代&#xff0c;从电商商品图自动打标&#xff0c;到社交媒体内容审核&#xff0c;再到智能客服中的图文问答&#xff0c;多模态AI能力正迅速成为各类应用的标配。然而&#xff0c;现实却…

作者头像 李华
网站建设 2026/5/4 15:41:55

GitHub镜像网站推荐2024:高效获取GLM-4.6V-Flash-WEB资源

GitHub镜像网站推荐2024&#xff1a;高效获取GLM-4.6V-Flash-WEB资源 在AI技术加速落地的今天&#xff0c;一个现实问题始终困扰着国内开发者——如何快速、稳定地获取开源大模型&#xff1f;尤其是当项目依赖动辄数GB的权重文件时&#xff0c;直接从GitHub拉取常常面临超时、…

作者头像 李华
网站建设 2026/5/3 12:20:52

Chromedriver下载地址总是失效?GLM-4.6V-Flash-WEB识别有效链接

Chromedriver下载地址总是失效&#xff1f;GLM-4.6V-Flash-WEB识别有效链接 在自动化测试和爬虫开发的世界里&#xff0c;一个看似微不足道的问题却可能让整个流程卡住——Chromedriver 下载链接 404 了。 这几乎是每个开发者都经历过的“日常噩梦”&#xff1a;CI/CD 流水线突…

作者头像 李华
网站建设 2026/5/3 10:26:33

GLM-4.6V-Flash-WEB模型深度解析:高并发场景下的视觉理解利器

GLM-4.6V-Flash-WEB模型深度解析&#xff1a;高并发场景下的视觉理解利器 在当今智能应用快速渗透的背景下&#xff0c;用户对系统的“看得懂、答得快”能力提出了前所未有的要求。无论是电商平台上传商品瑕疵图后自动判断责任归属&#xff0c;还是政务系统中上传表单即可获取…

作者头像 李华
网站建设 2026/5/3 9:44:36

Dify内容生成卡顿?快速定位并解决描述生成限速问题

第一章&#xff1a;Dify内容生成卡顿&#xff1f;快速定位并解决描述生成限速问题在使用 Dify 构建 AI 应用时&#xff0c;部分用户反馈在调用内容生成功能时出现明显延迟或“卡顿”现象。此类问题通常源于系统对生成请求的限速机制&#xff0c;尤其是在高并发或频繁调用场景下…

作者头像 李华
网站建设 2026/4/20 10:33:40

HTML表单上传图片交由GLM-4.6V-Flash-WEB进行云端分析

HTML表单上传图片交由GLM-4.6V-Flash-WEB进行云端分析 在今天的Web应用开发中&#xff0c;一个越来越常见的需求是&#xff1a;让用户像提问一样理解一张照片。比如&#xff0c;上传一张厨房台面的照片&#xff0c;问“我还能做哪些菜&#xff1f;”&#xff1b;或是拍下一段电…

作者头像 李华