热门标签 | HotTags
当前位置:  开发笔记 > 后端 > 正文

三个线程交替按顺序打印ABC之条件队列的理解

如题。本文给出交替打印的代码示例,并解释了条件变量在代码实现中所起的作用。使用三个线程,一个只负责打印A,另一个只负责打印B,最后一个只负责打印C按顺序交替。即打印A后,才能打印B

如题。本文给出交替打印的代码示例,并解释了条件变量在代码实现中所起的作用。



  • 使用三个线程,一个只负责打印A,另一个只负责打印B,最后一个只负责打印C

  • 按顺序交替。即打印A后,才能打印B,打印B后,才能打印C

由于按序交替,最好采用条件队列来实现。初始时,只有打印A的条件满足 打印B、C的条件都不满足。A打印后,使得打印B的条件满足,同时打印A的条件由原来的满足变成不满足;B打印后,使得打印C的条件满足,同时打印B的条件由原来的满足变成不满足;C打印后,使得打印A的条件满足,同时打印C的条件由原来的满足变成不满足。

采用锁+条件队列实现的优势:

锁+条件队列是基于"通知-唤醒"机制实现的,比sleep+轮询的方式要高效。这篇文章最后第6点简要说明了这2种机制。

完整代码如下:

import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.ReentrantLock;
/**
* @author psj
* @date 20-3-7
*/
public class PrintABC {
private ReentrantLock lock = new ReentrantLock();
//与锁关联的条件队列,当打印条件不满足时,挂起线程(通知唤醒机制,而不是sleep或者轮询)
private Condition printA = lock.newCondition();
private Condition printB = lock.newCondition();
private Condition printC = lock.newCondition();
//初始化 打印A的条件成立,打印B不成立,打印C不成立
private volatile boolean isA = true;
private volatile boolean isB = false;
private volatile boolean isC = false;
public static void main(String[] args) {
PrintABC pabc = new PrintABC();
Thread t1 = new Thread(new Runnable() {
@Override
public void run() {
while (true) {
try {
pabc.printA();
} catch (InterruptedException e) {
System.out.println(Thread.currentThread().getName() + " 退出打印");
break;
}
}
}
}, "t1");
Thread t2 = new Thread(new Runnable() {
@Override
public void run() {
while (true) {
try {
pabc.printB();
} catch (InterruptedException e) {
//响应中断退出打印
System.out.println(Thread.currentThread().getName() + " 退出打印");
break;
}
}
}
}, "t2");
Thread t3 = new Thread(new Runnable() {
@Override
public void run() {
while (true) {
try {
pabc.printC();
} catch (InterruptedException e) {
System.out.println(Thread.currentThread().getName() + " 退出打印");
break;
}
}
}
}, "t3");
t2.start();
t3.start();
t1.start();
// sleepMills(10 * 1000);
// t1.interrupt();
}
public void printA() throws InterruptedException{
try {
lock.lock();
while (!isA) {
printA.await();
}
System.out.println(Thread.currentThread().getName() + " print A");
sleepMills(2000);
//A 已打印,将打印A的条件由原来的满足变成不满足
isA = false;
//将打印B的条件变成满足
isB = true;
//通知线程打印B
printB.signal();
}finally {
lock.unlock();
}
}
public void printB()throws InterruptedException {
try {
lock.lock();
while (!isB) {
printB.await();
}
System.out.println(Thread.currentThread().getName() + " print B");
//模拟方法执行耗时
sleepMills(2000);
//打印B的条件由满足变成不满足
isB = false;
//使得打印C的条件变成满足
isC = true;
printC.signal();
}finally {
lock.unlock();
}
}
public void printC()throws InterruptedException {
try {
lock.lock();
while (!isC) {
printC.await();
}
System.out.println(Thread.currentThread().getName() + " print C");
sleepMills(2000);
//C已打印,将打印C的条件由原来的满足变成不满足
isC = false;
//将打印A的条件变成满足
isA = true;
printA.signal();
}finally {
lock.unlock();
}
}
private static void sleepMills(long mills) {
try {
TimeUnit.MILLISECONDS.sleep(mills);
} catch (InterruptedException e) {
System.out.println(e);
}
}
}

再来看一个交替打印AB的示例。这里给出了2种实现思路,一种是基于 volatile变量;另一种是采用条件队列。对比了这2种实现之后,讨论了条件队列背后的原理(通知唤醒机制、线程调度、线程阻塞状态……)

import java.util.concurrent.TimeUnit;
/**
* @author psj
* @date 20-3-7
*/
public class PrintAB {
private volatile boolean isA = true;
public static void main(String[] args) {
PrintAB pab = new PrintAB();
Thread t1 = new Thread(() -> {
while (true) {
pab.printA();
}
}, "t1");
Thread t2 = new Thread(() -> {
while (true) {
pab.printB();
}
}, "t2");
t2.start();
t1.start();
}
public void printA() {
if (isA) {
System.out.println(Thread.currentThread().getName() + " print A");
//模拟方法执行耗时
sleepMills(1000);
isA = false;
}
}
public void printB() {
if (!isA) {
System.out.println(Thread.currentThread().getName() + " print B");
sleepMills(2000);
isA = true;
}
}
private static void sleepMills(long mills) {
try {
TimeUnit.MILLISECONDS.sleep(mills);
} catch (InterruptedException e) {
System.out.println(e);
}
}
}

使用一个volatile变量协调2个线程交替打印A、B的顺序。此种方式是很消耗CPU的,因为:2个线程是在while true循环中不停地测试打印条件是否成立。另一种优雅的方式则是采用通知唤醒机制:当条件不成立时,让线程放弃cpu,挂起线程,进入阻塞状态(WAITING),当条件成立后,再唤醒线程,让它再次去争抢cpu,执行打印。这可以通过条件队列来实现,代码如下:

import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
/**
* @author psj
* @date 20-3-7
*/
public class PrintABCondition {
private Lock lock = new ReentrantLock();
private Condition pac = lock.newCondition();
private Condition pbc = lock.newCondition();
//决定打印A or 打印B 条件是否满足
private volatile boolean printA = true;
public void printA() throws InterruptedException{
try {
lock.lock();
while (!printA) {
//打印A的条件未满足,挂起线程,放弃cpu,进入WAITING状态
pac.await();
}
//打印A的条件满足了,打印A
System.out.println(Thread.currentThread().getName() + " print A");
//模拟方法执行耗时
sleepMills(1500);
//A 已经打印完毕, 使得打印B的条件满足, 接下来发送通知 唤醒打印B的线程
printA = false;
pbc.signal();
}finally {
lock.unlock();
}
}
public void printB() throws InterruptedException{
try {
lock.lock();
while (printA) {
//打印B的条件未满足,挂起线程,放弃cpu,进入WAITING状态
pbc.await();
}
System.out.println(Thread.currentThread().getName() + " print B");
sleepMills(2000);
//B 已打印完毕,使得打印A的条件满足,接下来发送通知 唤醒打印A的线程
printA = true;
pac.signal();
}finally {
lock.unlock();
}
}
public static void main(String[] args) {
PrintABCondition pab = new PrintABCondition();
Thread t1 = new Thread(() -> {
while (true) {
try {
pab.printA();
} catch (InterruptedException e) {
//响应中断
break;
}
}
}, "t1");
Thread t2 = new Thread(() -> {
while (true) {
try {
pab.printB();
} catch (InterruptedException e) {
break;
}
}
}, "t2");
t2.start();
t1.start();
}
private static void sleepMills(long mills) {
try {
TimeUnit.MILLISECONDS.sleep(mills);
} catch (InterruptedException e) {
System.out.println(e);
}
}
}

看juc并发包Condition.java的await方法里面有一段注释:


In all cases, before this method can return the current thread must re-acquire the lock associated with this condition。When the thread returns it is guaranteed to hold this lock.


这里从打印A的线程角度来解释一下:打印A的线程在从 await()方法返回时,必须重新争抢锁,争抢到锁之后,就会再执行while循环测试条件是否满足,如果此时条件满足(printA变为true)了,那就往下执行。如果条件不满足(printA为false),那么就放弃cpu,进入WAITING状态,等待唤醒。

从线程调度的角度来说,当执行Thread#start()后,线程从NEW状态变成RUNNABLE状态,此时线程具有运行的资格--可以被线程调度器选中占用cpu执行,但并不是说该线程一定占有cpu在运行了。由于"最小时间片"原则,每个线程一般都会占用cpu运行一小段时间,然后由于"抢占式调度",就被调度器切换出去了,线程不再占有cpu了(这种情形下的切换是多线程并发执行所固有的性质),与 "多个线程争抢同一把锁,未获得锁的线程被阻塞挂起,从而不再占有cpu了" 是不同的,要注意区分。

这里说一下为什么要在while循环里面测试条件,当条件不满足时,调用await方法使得线程放弃cpu,进入WAITING状态。为什么用while,if语句不可以吗?

我觉得用while循环的原因是:其它线程可能“无意”间调用了singal()使得该线程被唤醒了(又或者是线程因为某种未知原因唤醒了),线程醒来之后需要重新测试条件是否满足,所以只能用while循环。

实际上,await()底层是调用LockSupport#park(java.lang.Object)来挂起线程的,那看看该方法的注释,想起一个问题:当一个线程被阻塞挂起时,有哪些方法可以让它恢复执行?在开始讨论之前,再次明确一下:所谓恢复执行,只是使得线程"醒过来"具有执行的资格,并不一定保证线程就拿到了cpu,正在运行了,记住:抢占式调度,是由线程调度器来决定将哪个cpu分配给线程运行的。

OK,我觉得主要有两种方式唤醒线程,恢复执行。一种是"中断",即线程通过响应 InterruptedException 异常,退出阻塞状态;另一种是其它线程发送"通知",比如调用signal/signalAll方法(底层是调用LockSupport#unpark),使得线程退出阻塞状态。

但是,看LockSupport#park方法的注释,还提到了一种情况:


The call spuriously (that is, for no reason) returns.


这句话也验证了,为什么只能用while循环(不能用if语句)来测试条件是否满足(比如打印AB示例代码中的 printA 条件变量)的一个原因,因为线程可能不知道什么原因被唤醒了,只有while循环才能保证线程醒来之后会重新测试条件是否满足。

额外补充一下,这里为什么是线程阻塞后,是WAITING状态,而不是BLOCKED状态呢?哈哈。看 Thread.java 类的关于线程状态描述的源码注释(hint:等待条件满足)就知道了。

使用条件队列的好处:



  • 通知唤醒机制,代码高效

  • 能清楚看到线程在哪个条件上阻塞,并发逻辑清晰

参考资料:



  • 《JAVA并发编程实战》第14章 条件队列

  • 谈谈多线程

    原文:https://www.cnblogs.com/hapjin/p/12432928.html



推荐阅读
  • 本文介绍了Java并发库中的阻塞队列(BlockingQueue)及其典型应用场景。通过具体实例,展示了如何利用LinkedBlockingQueue实现线程间高效、安全的数据传递,并结合线程池和原子类优化性能。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 微软Exchange服务器遭遇2022年版“千年虫”漏洞
    微软Exchange服务器在新年伊始遭遇了一个类似于‘千年虫’的日期处理漏洞,导致邮件传输受阻。该问题主要影响配置了FIP-FS恶意软件引擎的Exchange 2016和2019版本。 ... [详细]
  • 探讨如何真正掌握Java EE,包括所需技能、工具和实践经验。资深软件教学总监李刚分享了对毕业生简历中常见问题的看法,并提供了详尽的标准。 ... [详细]
  • 作者:守望者1028链接:https:www.nowcoder.comdiscuss55353来源:牛客网面试高频题:校招过程中参考过牛客诸位大佬的面经,但是具体哪一块是参考谁的我 ... [详细]
  • 深入解析TCP/IP五层协议
    本文详细介绍了TCP/IP五层协议模型,包括物理层、数据链路层、网络层、传输层和应用层。每层的功能及其相互关系将被逐一解释,帮助读者理解互联网通信的原理。此外,还特别讨论了UDP和TCP协议的特点以及三次握手、四次挥手的过程。 ... [详细]
  • FinOps 与 Serverless 的结合:破解云成本难题
    本文探讨了如何通过 FinOps 实践优化 Serverless 应用的成本管理,提出了首个 Serverless 函数总成本估计模型,并分享了多种有效的成本优化策略。 ... [详细]
  • 本文详细探讨了HTML表单中GET和POST请求的区别,包括它们的工作原理、数据传输方式、安全性及适用场景。同时,通过实例展示了如何在Servlet中处理这两种请求。 ... [详细]
  • 深入解析Redis内存对象模型
    本文详细介绍了Redis内存对象模型的关键知识点,包括内存统计、内存分配、数据存储细节及优化策略。通过实际案例和专业分析,帮助读者全面理解Redis内存管理机制。 ... [详细]
  • 探讨如何通过高效的数据库查询和排序策略,优化基于GPS位置信息的附近用户搜索功能,以应对大规模用户数据场景。 ... [详细]
  • 本文探讨了哪些数据库支持队列式的写入操作(即一个键对应一个队列,数据可以连续入队),并且具备良好的持久化特性。这类需求通常出现在需要高效处理和存储大量有序数据的场景中。 ... [详细]
  • Netflix利用Druid实现高效实时数据分析
    本文探讨了全球领先的在线娱乐公司Netflix如何通过采用Apache Druid,实现了高效的数据采集、处理和实时分析,从而显著提升了用户体验和业务决策的准确性。文章详细介绍了Netflix在系统架构、数据摄取、管理和查询方面的实践,并展示了Druid在大规模数据处理中的卓越性能。 ... [详细]
  • 深入解析for与foreach遍历集合时的性能差异
    本文将详细探讨for循环和foreach(迭代器)在遍历集合时的性能差异,并通过实际代码示例和源码分析,帮助读者理解这两种遍历方式的不同之处。文章内容丰富且专业,旨在为编程爱好者提供有价值的参考。 ... [详细]
  • 全面解析运维监控:白盒与黑盒监控及四大黄金指标
    本文深入探讨了白盒和黑盒监控的概念,以及它们在系统监控中的应用。通过详细分析基础监控和业务监控的不同采集方法,结合四个黄金指标的解读,帮助读者更好地理解和实施有效的监控策略。 ... [详细]
  • 本文详细介绍了Grand Central Dispatch (GCD) 的核心概念和使用方法,探讨了任务队列、同步与异步执行以及常见的死锁问题。通过具体示例和代码片段,帮助开发者更好地理解和应用GCD进行多线程开发。 ... [详细]
author-avatar
丽春院少爷
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有