[Java Concurrent] 多线程合作 wait / notifyAll 的简单案例
本案例描述的是,给一辆汽车打蜡、抛光的场景。
Car 是一辆被打蜡抛光的汽车,扮演共享资源的角色。
WaxOnCommand 负责给汽车打蜡,打蜡时需要独占整部车,一次打一部分蜡,等待抛光,然后再打一部分蜡。
BuffCommand 负责给汽车抛光,抛光时需要独占整部车,每一次会将刚打上的蜡抛光,然后等待打蜡,然后再将刚打上的蜡抛光。
WaxOnCommand 、BuffCommand 分别由两个线程相互交替地执行:
WaxOnCommand 打蜡、等待抛光,BuffCommand 抛光,等待打蜡;
WaxOnCommand 打蜡、等待抛光,BuffCommand 抛光,等待打蜡;
WaxOnCommand 打蜡、等待抛光,BuffCommand 抛光,等待打蜡;
......
CarWaxBuffDemo 演示这个场景。
下面是类之间的关系图。
由于 ExecutorServices 的使用和 Command Pattern 很相似,所以参照其 UML 绘制而成。
代码实现:
Car, 可以改变汽车的打蜡状态,可以等待打蜡状态的改变。输出内容到控制台时,会把当前执行线程一一同输出。
public class Car { private boolean isWaxed = false; public synchronized void waxed(){
isWaxed = true;
print(" car is waxed now ");
notifyAll();
} public synchronized void unwaxed(){
isWaxed = false;
print(" car is unwaxed now ");
notifyAll();
}public synchronized void waitingForReWax() throws InterruptedException{
while (isWaxed == true){
wait();
}
print(" OK to re-wax "); } public synchronized void waitingForReBuff() throws InterruptedException{
while (isWaxed == false){
wait();
}
print(" OK to re-Buff ");
} public void print(String msg){
System.out.println("[" + Thread.currentThread().getName() + " ] " + msg);
}
}
WaxonComman,给汽车打蜡,主要是两个动作:1.等待可以打蜡,2.打蜡。
import java.util.concurrent.TimeUnit; public class WaxonCommand implements Runnable { private Car car; public WaxonCommand(Car car){
this.car = car;
} @Override
public void run() {
try {
while (true){
car.waitingForReWax(); print(" WaxOn take action ");
TimeUnit.MILLISECONDS.sleep(); car.waxed();
}
}catch (Exception e) {
e.printStackTrace();
} try {
TimeUnit.MILLISECONDS.sleep();
} catch (InterruptedException e) {
e.printStackTrace();
} print(" Ending WaxOn action ");
} public void print(String msg){
System.out.println("[" + Thread.currentThread().getName() + " ] " + msg);
}
}
buffCommand, 给汽车抛光,主要两个动作:1.等待可以抛光,2. 抛光。
import java.util.concurrent.TimeUnit; public class BuffCommand implements Runnable{ private Car car; public BuffCommand(Car car){
this.car = car;
} @Override
public void run() {
try {
while (true){
car.waitingForReBuff(); print(" Buff take action ");
TimeUnit.MILLISECONDS.sleep(); car.unwaxed();
}
}catch (InterruptedException e) {
e.printStackTrace();
} try {
TimeUnit.MILLISECONDS.sleep();
} catch (InterruptedException e) {
e.printStackTrace();
} print(" Ending Buff action "); } public void print(String msg){
System.out.println("[" + Thread.currentThread().getName() + " ] " + msg);
}
}
演示给汽车交替打蜡、抛光的场景。给 1 秒的时间,让 WaxonCommand / BuffComand 线程进行打蜡、抛光操作,1 秒后,中断他们的操作。
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit; public class CarWaxBuffDemo { public static void main() throws InterruptedException{
ExecutorService exec = Executors.newCachedThreadPool(); Car car = new Car();
exec.execute(new BuffCommand(car));
exec.execute(new WaxonCommand(car)); TimeUnit.SECONDS.sleep(1);
System.out.println(" executor stopping ");
exec.shutdownNow();
}
}
输出结果:
在被强制中断前,thread-1 和 thread-2 交替进行打蜡、抛光操作。
[pool-1-thread-1 ] OK to re-wax
[pool-1-thread-1 ] WaxOn take action
[pool-1-thread-1 ] car is waxed now
[pool-1-thread-2 ] OK to re-Buff
[pool-1-thread-2 ] Buff take action
[pool-1-thread-2 ] car is unwaxed now
[pool-1-thread-1 ] OK to re-wax
[pool-1-thread-1 ] WaxOn take action
[pool-1-thread-1 ] car is waxed now
[pool-1-thread-2 ] OK to re-Buff
[pool-1-thread-2 ] Buff take action
[pool-1-thread-2 ] car is unwaxed now
[pool-1-thread-1 ] OK to re-wax
[pool-1-thread-1 ] WaxOn take action
[pool-1-thread-1 ] car is waxed now
[pool-1-thread-2 ] OK to re-Buff
[pool-1-thread-2 ] Buff take action
[pool-1-thread-2 ] car is unwaxed now
[pool-1-thread-1 ] OK to re-wax
[pool-1-thread-1 ] WaxOn take action
[pool-1-thread-1 ] car is waxed now
[pool-1-thread-2 ] OK to re-Buff
[pool-1-thread-2 ] Buff take action
[pool-1-thread-2 ] car is unwaxed now
[pool-1-thread-1 ] OK to re-wax
[pool-1-thread-1 ] WaxOn take action
[pool-1-thread-1 ] car is waxed now
[pool-1-thread-2 ] OK to re-Buff
[pool-1-thread-2 ] Buff take action
[pool-1-thread-2 ] car is unwaxed now
[pool-1-thread-1 ] OK to re-wax
[pool-1-thread-1 ] WaxOn take action
executor stopping
java.lang.InterruptedException
at java.lang.Object.wait(Native Method)
at java.lang.Object.wait(Object.java:502)
at concurrencyCooperation.Car.waitingForReBuff(Car.java:29)
at concurrencyCooperation.BuffCommand.run(BuffCommand.java:17)
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)
at java.lang.Thread.run(Thread.java:745)
java.lang.InterruptedException: sleep interrupted
at java.lang.Thread.sleep(Native Method)
at java.lang.Thread.sleep(Thread.java:340)
at java.util.concurrent.TimeUnit.sleep(TimeUnit.java:386)
at concurrencyCooperation.WaxonCommand.run(WaxonCommand.java:20)
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)
at java.lang.Thread.run(Thread.java:745)
[pool-1-thread-2 ] Ending Buff action
[pool-1-thread-1 ] Ending WaxOn action
参考资料 :
Page 857, wait() and notifyAll(), Thinking in Java
[Java Concurrent] 多线程合作 wait / notifyAll 的简单案例的更多相关文章
- [Java Concurrent] 多线程合作 producer-consumers / queue 的简单案例
在多线程环境下,通过 BlockingQueue,实现生产者-消费者场景. Toast 被生产和消费的对象. ToastQueue 继承了 LinkedblockingQueue ,用于中间存储 To ...
- java基础之Socket编程概述以及简单案例
概述: 用来实现网络互连的 不同的计算机上 运行的程序间 可以进行数据交互 也就是用来在不同的电脑间, 进行数据传输. 三大要素: IP地址: 设备(电脑,手机,ipad)在网络中的唯一标识. 组成 ...
- Java Tread多线程(0)一个简单的多线程实例
作者 : 卿笃军 原文地址:http://blog.csdn.net/qingdujun/article/details/39341887 本文演示,一个简单的多线程实例,并简单分析一下线程. 编程多 ...
- Java Tread多线程(1)实现Runnable接口
作者 : 卿笃军 原文地址:http://blog.csdn.net/qingdujun/article/details/39347245 本文演示,Tread多线程实现Runnable接口,以及简单 ...
- Java:多线程,java.util.concurrent.atomic包之AtomicInteger/AtomicLong用法
1. 背景 java.util.concurrent.atomic这个包是非常实用,解决了我们以前自己写一个同步方法来实现类似于自增长字段的问题. 在Java语言中,增量操作符(++)不是原子的,也就 ...
- Java的多线程 简单入门
Java的多线程 简单入门 首先能够先搞清楚什么是程序.进程.线程,以及它们之间的关系: 定义: 一 程序仅仅是一组指令的有序集合.它是静态的 二 进程是具有一定独立功能的程序关于某个数据集合上的一次 ...
- Java的多线程机制系列:(一)总述及基础概念
前言 这一系列多线程的文章,一方面是个人对Java现有的多线程机制的学习和记录,另一方面是希望能给不熟悉Java多线程机制.或有一定基础但理解还不够深的读者一个比较全面的介绍,旨在使读者对Java的多 ...
- java 并发多线程显式锁概念简介 什么是显式锁 多线程下篇(一)
目前对于同步,仅仅介绍了一个关键字synchronized,可以用于保证线程同步的原子性.可见性.有序性 对于synchronized关键字,对于静态方法默认是以该类的class对象作为锁,对于实例方 ...
- Java的多线程实现生产/消费模式
Java的多线程实现生产/消费模式 在Java的多线程中,我们经常使用某个Java对象的wait(),notify()以及notifyAll() 方法实现多线程的通讯,今天就使用Java的多线程实现生 ...
随机推荐
- yii columns value and type and checkbox columns
value I am here type I am here checkbox columns useage
- 【javascript模式】Chapter2: 基本 技巧
1 尽量少用全局变量,最好一个应用程式只有一个全局变量 隐含全局变量(不使用var声明)与明确定义的全局变量区别: (1)使用var创建的全局变量(在函数外部声明)不能用delete删除 (2) ...
- 【转】iOS6中的Auto Layout:通过代码添加约束
最近做的项目用到了Auto Layout,于是经过了一番大量的google,这是我看到的讲用代码创建约束最清晰的一篇教程,于是想跟更多的人分享一下.原文也比较简单,可以直接过去看,如果我翻译的 ...
- MTP设备无法安装驱动的解决办法
1,进入设备管理器右击带黄色问号的MTP,选择“属性”,“详细信息”“设备范例 ID”(用Ctrl+C复制). 2,找到c:\windows\inf\wpdmtp.inf打开(或者通过运行打开),找到 ...
- 动态内存分配(new)和释放(delete)
在之前我们所写过的程序中,所必需的内存空间的大小都是在程序执行之前就已经确定了.但如果我们需要内存大小为一个变量,其数值只有在程序运行时 (runtime)才能确定,例如有些情况下我们需要根据用户输入 ...
- js关于DOM和BOM
关于BOM和DOM BOM 下面一幅图很好的说明了BOM和DOM的关系 BOM提供了一些访问窗口对象的一些方法,我们可以用它来移动窗口位置,改变窗口大小,打开新窗口和关闭窗口,弹出对话框,进行导航以及 ...
- 获取当前元素节点的position和宽高(兼容)
function objxy(n){ var o=document.getElementById(n),x=0,y=0,w=o.offsetWidth,h=o.offsetHeight if(o.ge ...
- highcharts实例教程一:结合php与mysql生成折线图
Highcharts是一款纯javascript和html5编写的图表库,不仅几乎能兼容所有pc浏览器,而且对ios和android手机端的兼容 性也不错,它能够很简单便捷的在Web网站或Web应用中 ...
- Sql 格式化工具
SQL Pretty Printer:目前提供4种使用方式,桌面版本,SSMS(SQL Server Management Studio)插件,VS插件,和提供API接口. SQL Pretty Pr ...
- python百科
Python 编辑词条 添加义项名 B 添加义项 ? Python(英语发音:/ˈpaɪθən/), 是一种面向对象.解释型计算机程序设计语言,由Guido van Rossum于1989年底发明,第 ...