[图解Java]Condition
图解Condition
0. demo
我先给出一个demo, 这样大家就可以根据我给的这段代码, 边调试边看源码了. 还是那句话: 注意"My" , 我把ReentrantLock类 改名为了 "MyReentrantLock"类 , "Lock"类 改名为了"MyLock"类. 大家粘贴我的代码的时候, 把相应的"My"都去掉就好了, 否则会编译报错哦.
import java.util.Scanner;
import java.util.concurrent.locks.Condition;
import java.util.function.Supplier; public class ConditionTest {
static final Scanner scanner = new Scanner(System.in);
static volatile String cmd = "";
private static MyReentrantLock lock = new MyReentrantLock(true);
private static Condition condition = lock.newCondition(); public static void main(String[] args) {
for (String name : new String[]{"w1", "w2", "w3", "w4", "w5", "w6"})
new Thread(() -> func(() -> lock, name)).start();
new Thread(() -> signalOne(() -> lock, "s")).start(); while (scanner.hasNext()) {
cmd = scanner.nextLine();
}
} public static void func(Supplier<MyLock> myLockSupplier, String name) {
blockUntilEquals(() -> cmd, name);
myLockSupplier.get().lock(); System.out.println(name + "阻塞等待...");
try {
condition.await();
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("释放了" + name); myLockSupplier.get().unlock();
} public static void signalOne(Supplier<MyLock> myLockSupplier, String name) {
while (true) {
blockUntilEquals(() -> cmd, name);
myLockSupplier.get().lock();
condition.signal();
System.out.println("通知唤醒了一个等待...");
myLockSupplier.get().unlock();
}
} private static void blockUntilEquals(Supplier<String> cmdSupplier, final String expect) {
while (!cmdSupplier.get().equals(expect))
quietSleep(1000);
clearCmd();
} private static void quietSleep(int mills) {
try {
Thread.sleep(mills);
} catch (InterruptedException e) {
e.printStackTrace();
}
} private static void clearCmd() {
cmd = "";
}
}
使用例子在下面.
首先输入w1, 让线程1执行await() . 然后输入w2, 让线程2执行await(). 然后输入w3, 让线程3执行await().
接下来输入3次 s, 没输入一次s, 并按下回车, 就会signal通知一个await等待.
1. 开始图解Condition
想用ReentrantLock的Condition, 那么就首先要有个ReentrantLock锁.
实例化一个锁, ReentrantLock里只有一个成员变量sync.
sync实例里面有四个成员变量.
分别表示:
1. state - 锁计数器
2. exclusiveOwnerThread - 锁的持有线程
3. head - `等待队列`的头结点.
4. tail - 指向`等待队列`的最后一个元素
然后咱们实例化了一个Condition.
当咱们输入w1后, 第一个线程就申请了锁, 并且申请成功.
然后就执行到了await()方法.
将线程1封装为Node节点, 然后waitState置为-2. -2的含义是Condition.
await()方法内部的第一个步骤就是把当前线程(线程1)插入到了`条件队列`中.
然后就开始释放当前线程(线程1)的锁了, 而且是完全释放, 一次就释放掉全部重入次数哦, 也就是直接让state等于0.
释放完锁了, 然后挂起线程1.
然后让线程2进行await.( 也就是前面的demo程序中在控制台输入了w2.)
线程2执行await()之前当然是先获取锁了.
由于此时, 锁是空闲的. 所以线程2成功获取到了锁. 淡橙色的阴影部分为变化的内容:
获取锁之后, 线程2就该执行await()了:
如上图, 将线程封装为Node, 然后尾插到`条件队里`中, 只是await() 方法的第一步.
然后的操作, 就是完全释放线程2的锁, 然后挂起线程.
如果这个时候咱们在上面demo程序的控制台输入"s", 那么就会让线程s 申请锁, 申请成功后, 就会执行signal.
首先是线程s申请锁成功:
线程s成功获取了锁以后, 就是该执行signal()了.
首先将`条件队列`里的第一个节点脱离出来:
然后把waitState从-2改为0 :
随后要做的就是把从`条件队列`中脱离出来的Node(就是线程1对应的Node节点), 尾插到`等待队列`中.
但是`等待队列`此时还未被初始化, 所以插入到`等待队列`之前, 要把`等待队列`初始化了. 见下图:
`等待队列`初始化完了. 接下来就是把线程1对应的Node, 尾插到`等待队列`中了:
然后将当前尾插的那个节点的前驱的waitState置为-1. -1表示下一个节点等待着被唤醒.
接下来就是线程s会执行到unlock(). 然后就会释放锁, 之后就是唤醒`等待队列`中的第一个线程.
如此就介绍完了signal()
一个signal命令, 就把一个await的线程从`条件队列中`移到了`等待队列`中. 到了等待队列中之后, 剩下的就是跟"锁解锁后, 唤醒下一个执行"这样的步骤一样了.
我觉得await方法就是将线程尾插到`条件队列`中. signal()方法就是把条件队列中的第一个元素, 尾插入到`等待队列`中.
所以我觉得不必往下分析了.
也可能是我理所当然了, 有问题的话之后再补充.
[图解Java]Condition的更多相关文章
- 图解 Java IO : 二、FilenameFilter源码
Writer :BYSocket(泥沙砖瓦浆木匠) 微 博:BYSocket 豆 瓣:BYSocket FaceBook:BYSocket Twitter ...
- 图解 Java 内存模型
图解 Java 内存模型 (图片来自于:http://www.cnblogs.com/zhangs1986/p/7903722.html)
- [温故]图解java多线程设计模式(一)
去年看完的<图解java多线程设计模式>,可惜当时没做笔记,导致后来忘了许多东西,打算再温习下这本书,顺便在这里记录一下~ 1.顺序执行.并行.并发 顺序执行:多个操作按照顺序依次执行. ...
- 【转载】图解Java常用数据结构(一)
图解Java常用数据结构(一) 作者:大道方圆 原文:https://www.cnblogs.com/xdecode/p/9321848.html 最近在整理数据结构方面的知识, 系统化看了下Jav ...
- 图解java面试
图解Java面试题:基本语法 2017-02-07 14:34 出处:清屏网 人气:178 评论(0) 内容大纲.png &和&&的区别 &和&&的 ...
- 图解Java继承内存分配
图解Java继承内存分配 继承的基本概念: (1)Java不支持多继承,也就是说子类至多只能有一个父类. (2)子类继承了其父类中不是私有的成员变量和成员方法,作为自己的成员变量和方法. (3)子 ...
- (转载)图解Java多态内存分配以及多态中成员方法的特点
图解Java多态内存分配以及多态中成员方法的特点 图解Java多态内存分配以及多态中成员方法的特点 Person worker = new Worker(); 子类实例对象地址赋值给父类类型引 ...
- 图解Java设计模式之UML类图
图解Java设计模式之UML类图 3.1 UML基本介绍 UML图 UML类图 3.1 UML基本介绍 1)UML – Unified modeling language UML(统一建模语言),是一 ...
- 图解Java设计模式之设计模式七大原则
图解Java设计模式之设计模式七大原则 2.1 设计模式的目的 2.2 设计模式七大原则 2.3 单一职责原则 2.3.1 基本介绍 2.3.2 应用实例 2.4 接口隔离原则(Interface S ...
随机推荐
- selenium-webdriver的二次封装(十)
接着上篇随笔 selenium-配置文件定位元素 ,进行了配置文件设置后,将配置文件运用到定位元素中 思路:拿到定位的 key 和 value 后,对 webdrvier 中定位进行封装,使可以直接运 ...
- Git 密钥对处理
生成密钥对: ssh-keygen -t rsa cd .ssh ls id_rsa 私钥 id_rsa.pub 公钥
- 如何学好java
今天发现这么一篇文章,对于笔者谈的:"一方面很努力学习,一方面又觉得不踏实",我感同身受.觉得文章写得不错,在此献给一些在java中努力的朋友们,希望能有所收获. 文章原内容: 近 ...
- Rabbitmq集群高可用
转载:https://www.cnblogs.com/flat_peach/archive/2013/04/07/3004008.html RabbitMQ是用erlang开发的,集群非常方便,因为e ...
- Pycharm 常用快捷键
常用快捷键 快捷键 功能 Ctrl + Q 快速查看文档 Ctrl + F1 显示错误描述或警告信息 Ctrl + / 行注释(可选中多行) Ctrl + Alt + L 代码格式化 Ctrl + A ...
- win7系统IE浏览器主页被搜狗篡改问题的解决方法
IE浏览器使用一段时间后可能大家就会遇到主页被篡改的问题,篡改之后主页就变成了搜狗页面,我们常用的百度搜索也变成了搜狗搜索,这不仅使得我们操作起 来不习惯,使用起来也会感觉非常别扭.那如果在使用IE浏 ...
- master公式 ------ 求递归情况下的时间复杂度
剖析递归行为和递归行为时间复杂度的估算一个递归行为的例子T(N) = a*T(N/b) + O(N^d)1) log(b,a) > d -> 复杂度为O(N^log(b,a))2) log ...
- Git使用注意事项
第一次用git时push时,突然想到我没有设置ssh key,却也可以push代码到自己仓库,那我本地登陆的账号Git是存在哪儿了呢? Git本地账户凭证管理 在第一次push到远程仓库时,git会提 ...
- gradle配置国内镜像
对单个项目生效,在项目中的build.gradle修改内容 buildscript { repositories { maven { url 'http://maven.aliyun.com/nexu ...
- 【MySQL 读书笔记】SQL 刷脏页可能造成数据库抖动
开始今天读书笔记之前我觉得需要回顾一下当我们在更新一条数据的时候做了什么. 因为 WAL 技术的存在,所以当我们执行一条更新语句的时候是先写日志,后写磁盘的.当我们在内存中写入了 redolog 之后 ...