一、Condition类的signal()方法底层原理

Condition 接口的 signal 方法是用于唤醒一个在 Condition 上等待的线程。与 Object 的 notify 方法类似,signal 方法会从 Condition 的等待队列中选择一个线程并将其唤醒,使其重新尝试获取锁并继续执行

1、signal 方法的核心逻辑

signal 方法的主要逻辑可以分为以下几个步骤:

1、检查当前线程是否持有锁:调用 signal 的线程必须持有与 Condition 关联的锁。

2、从等待队列中移除第一个节点:将 Condition 等待队列中的第一个节点转移到 AQS 的同步队列中。

3、唤醒线程:通过 AQS 的机制唤醒转移后的线程,使其重新尝试获取锁。

2、signal 方法的源码分析

以下是 ConditionObject 中 signal 方法的源码及其详细分析:

public final void signal() {
// 判断调用 signal 方法的线程是否是独占锁持有线程
if (!isHeldExclusively())
throw new IllegalMonitorStateException(); // 获取条件队列中第一个 Node
Node first = firstWaiter; // 不为空就将该节点【迁移到阻塞队列】
if (first != null)
doSignal(first);
}

2.1、isHeldExclusively 方法

  • 检查当前线程是否独占持有与 Condition 关联的锁。

  • 如果当前线程未持有锁,抛出 IllegalMonitorStateException。

2.2、firstWaiter 变量

  • firstWaiter 是 Condition 等待队列的头节点。

  • 如果 firstWaiter 为 null,说明等待队列为空,没有需要唤醒的线程。

2.3、doSignal 方法

  • doSignal 是实际执行唤醒操作的方法。

  • 它会将等待队列中的第一个节点转移到 AQS 的同步队列中,并唤醒对应的线程。

3、doSignal 和 signalAll方法的源码分析

3.1、以下是 doSignal 方法的源码及其详细分析:

// 唤醒 - 【将第一个节点转移至 AQS 队列尾部】
private void doSignal(Node first) {
do {
// 成立说明当前节点的下一个节点是 null,当前节点是尾节点了,队列中只有当前一个节点了
if ((firstWaiter = first.nextWaiter) == null)
lastWaiter = null;
first.nextWaiter = null;
// 将 Condition条件队列中的 Node 转移至 AQS 队列,不成功且还有节点则继续循环
} while (!transferForSignal(first) && (first = firstWaiter) != null);
}
  • 将 firstWaiter 指向下一个节点。

  • 如果等待队列为空,将 lastWaiter 设置为 null。

  • 断开当前节点的链接,将其从Condition条件队列中移除。

3.2、会调用signalAll()这个函数,唤醒所有的节点

private void doSignalAll(Node first) {
lastWaiter = firstWaiter = null;
do {
Node next = first.nextWaiter;
first.nextWaiter = null;
transferForSignal(first);
first = next;
// 唤醒所有的节点,都放到阻塞队列中
} while (first != null);
}

4、transferForSignal 方法的源码分析

transferForSignal作用

  • transferForSignal 是将节点从等待队列转移到 AQS 同步队列的核心方法。

  • 如果转移成功,返回 true;否则返回 false。

以下是 transferForSignal 方法的源码及其详细分析:

// 如果节点状态是取消, 返回 false 表示转移失败, 否则转移成功
final boolean transferForSignal(Node node) {
// CAS 修改当前节点的状态,修改为 0,因为当前节点马上要迁移到阻塞队列了
// 如果状态已经不是 CONDITION, 说明线程被取消(await 释放全部锁失败)或者被中断(可打断 cancelAcquire)
if (!compareAndSetWaitStatus(node, Node.CONDITION, 0))
// 返回函数调用处继续寻找下一个节点
return false; // 【先改状态,再进行迁移】
// 将当前 node 加入 AQS阻塞队列,p 是当前节点在阻塞队列的【前驱节点】
Node p = enq(node);
int ws = p.waitStatus; // 如果前驱节点被取消或者不能设置状态为 Node.SIGNAL,就 unpark 取消当前节点线程的阻塞状态,
// 让 thread-0 线程竞争锁,重新同步状态
if (ws > 0 || !compareAndSetWaitStatus(p, ws, Node.SIGNAL))
LockSupport.unpark(node.thread);
return true;
}

4.1、更新节点状态

  • 使用 CAS 操作将节点的状态从 CONDITION 更新为 0。

  • 如果更新失败,说明节点已被取消,返回 false。

4.2、将节点加入 AQS同步队列

  • 调用 enq(node) 方法将节点加入到 AQS 的同步队列中。

  • enq 方法会将节点插入到同步队列的尾部,并返回前驱节点。

4.3、唤醒线程

  • 检查前驱节点的状态:

    • 如果前驱节点的状态为 CANCELLED,或者无法将其状态更新为 SIGNAL,则直接唤醒线程。

    • 否则,线程会在同步队列中等待,直到前驱节点释放锁。

二、signal 方法的关键点

1、线程安全性

  • signal 方法必须在持有锁的情况下调用,否则会抛出 IllegalMonitorStateException。

  • 通过 AQS 的同步队列机制,确保线程安全。

2、节点转移

  • signal 方法会将节点从 Condition 的等待队列转移到 AQS 的同步队列中。

  • 转移后的节点会等待获取锁,并在获取锁后继续执行。

3、唤醒机制

  • 使用 LockSupport.unpark 唤醒线程。

  • 唤醒的线程会重新尝试获取锁,并在获取锁后从 await 方法中返回。

三、总结

Condition 的 signal 方法通过以下机制实现线程的唤醒:

1、检查当前线程是否持有锁。

2、将等待队列中的第一个节点转移到 AQS 的同步队列中。

3、唤醒对应的线程,使其重新尝试获取锁。

Condition类的signal()方法底层原理的更多相关文章

  1. KVO-基本使用方法-底层原理探究-自定义KVO-对容器类的监听

    书读百变,其义自见! 将KVO形式以代码实现呈现,通俗易懂,更容易掌握 :GitHub   -链接如果失效请自动搜索:https://github.com/henusjj/KVO_base 代码中有详 ...

  2. String类中intern方法的原理分析

    一,前言 ​ 昨天简单整理了JVM内存分配和String类常用方法,遇到了String中的intern()方法.本来想一并总结起来,但是intern方法还涉及到JDK版本的问题,内容也相对较多,所以今 ...

  3. Swift-方法调度-类的普通方法底层探究

    1. 类的普通方法调度 写一个结构体和一个类,对比看看方法调用的方式: // 结构体 struct PersonStruct { func changClassName() {} } let s = ...

  4. java多线程,多线程加锁以及Condition类的使用

    看了网上非常多的运行代码,很多都是重复的再说一件事,可能对于java老鸟来说,理解java的多线程是非常容易的事情,但是对于我这样的菜鸟来说,这个实在有点难,可能是我太菜了,网上重复的陈述对于我理解这 ...

  5. synchronized底层原理

    synchronized底层语义原理 Java 虚拟机中的同步(Synchronization)基于进入和退出管程(Monitor)对象实现. 在 Java 语言中,同步用的最多的地方可能是被 syn ...

  6. 线程池底层原理详解与源码分析(补充部分---ScheduledThreadPoolExecutor类分析)

    [1]前言 本篇幅是对 线程池底层原理详解与源码分析  的补充,默认你已经看完了上一篇对ThreadPoolExecutor类有了足够的了解. [2]ScheduledThreadPoolExecut ...

  7. object的wait()、notify()、notifyAll()、方法和Condition的await()、signal()方法

    wait().notify()和notifyAll()是 Object类 中的方法 从这三个方法的文字描述可以知道以下几点信息: 1)wait().notify()和notifyAll()方法是本地方 ...

  8. 红黑树规则,TreeSet原理,HashSet特点,什么是哈希值,HashSet底层原理,Map集合特点,Map集合遍历方法

    ==学习目标== 1.能够了解红黑树 2.能够掌握HashSet集合的特点以及使用(特点以及使用,哈希表数据结构) 3.能够掌握Map集合的特点以及使用(特点,常见方法,Map集合的遍历) 4.能够掌 ...

  9. ArrayList类中的contains()方法底层依赖的是equals()方法

    ArrayList类中的contains()方法底层依赖的是equals()方法.若集合中的元素是自定义对象,则应该重写该类父类Object的equals()方法,否则对象永远都不相同(因为都是new ...

  10. HashMap底层原理分析(put、get方法)

    1.HashMap底层原理分析(put.get方法) HashMap底层是通过数组加链表的结构来实现的.HashMap通过计算key的hashCode来计算hash值,只要hashCode一样,那ha ...

随机推荐

  1. 自定义快捷命令程序(VC++加批处理)

    一 概述 在看<从小工到专家-程序员修炼之道>时,看到建议使用Shell,很有感触.在很多时候,通过键盘操作,比鼠标的确会块很多,如果能用好shell命令(或批处理命令)   ,的确能节省 ...

  2. try-catch-finally的使用

    /* * 一.异常的处理:抓抛模型 * * 过程一:"抛":程序在正常执行的过程中,一旦出现异常,就会在异常代码处生成一个对应异常类的对象. * 并将此对象抛出. * 一旦抛出对象 ...

  3. Educational Codeforces Round 172 (Rated for Div. 2)(C-D)

    题目链接:Dashboard - Educational Codeforces Round 172 (Rated for Div. 2) - Codeforces C. Competitive Fis ...

  4. Navicat怎样查看数据库密码

    Navicat怎样查看数据库密码 前言 本文来源:Navicat怎样查看数据库密码_低端玩家的博客-CSDN博客_navicat查看数据库密码 主要是怕作者删帖,因此备份 开始 1.导出链接 2.一定 ...

  5. 0101-win10 jkd配置注意事项

    更换新的电脑预装win10家庭版,根据常规方法配置jdk8后运行javac提示:不是内部或外部命令,也不是可运行的程序或批处理文件. 1 设置变量classpath时前面有个点(完成这一步后javac ...

  6. 用SignalR和Layui搭建自己的web聊天网站

    1.开发背景 之前是做项目一直有一个困扰,就是如何进行及时通讯,本人.Net开发,不太想用别人的接口,然后偶然的机会知道了SignalR,那么什么是SignalR呢? 2.SignalR简介 ASP. ...

  7. 一个 .NET 开源、免费、功能强大的Windows应用卸载神器

    前言 今天大姚给大家分享一个基于 .NET 开源(Apache License).免费.功能强大的Windows应用卸载神器:Bulk Crap Uninstaller. 项目介绍 Bulk Crap ...

  8. flutter-延时执行

    //1秒后这个i行 Future.delayed(Duration(milliseconds: 1000), () { //代码省略 });

  9. [JXOI2017] 加法 题解

    最小值最大,考虑二分答案,问题转为判断最小值是否能 \(\ge x\). 假如 \(a_i\ge x\),那我们肯定不管:假如 \(a_i<x\),那最好能让选择的区间 \(r\) 值更大,用优 ...

  10. [SDOI2008] Sandy的卡片 题解

    讲一种自认为最暴力的方法. 首先肯定还是用差分的思想,对于每一张卡片进行重新标号,在卡片串与卡片串中插入特殊字符,然后找重复了 \(n\) 次的子串. 这里我们对于每一个子串开一个大小为 \(n\) ...