线程通信【详解】

server/2024/10/18 23:27:01/

线程通信

线程通信是指线程间可以交互,指定信号,让线程执行或者等待

通过Object类中的方法完成通信

  • wait()

  • notify()

1、 同步方法

需求: 两个输出的方法,保证正常输出不被打断且达到一人一次输出的效果

java">public class Printer {
​// 定义一个信号量// 1代表print1执行 2代表print2执行private int flag = 1;
​/*** 线程通信的要求* 1) 要保证线程安全* 2) 线程等待方法是wait*    线程唤醒方法是notify* 3) 必须使用锁对象调用 通信的方法* ---------* 为什么,wait和notify这些线程通信的方法要设计在Object类?* 答:*/public synchronized void print1() throws InterruptedException {if (flag != 1){ // 信号不是1,说明不该print1执行,那就等待this.wait();}System.out.print(1 + " ");System.out.print(2 + " ");System.out.print(3 + " ");System.out.print(4 + " ");System.out.println( );
​// 改变信号量flag = 2;// 通知处于等待状态的线程启动this.notify();}public synchronized void print2() throws InterruptedException {if (flag != 2) { // 信号不是2,说明不该print2执行,那就等待this.wait();}System.out.print("A ");System.out.print("B ");System.out.print("C ");System.out.print("D ");System.out.println( );
​flag = 1;// 通知处于等待状态的线程启动this.notify();}
}
java">public class TestPrinter {
​public static void main(String[] args) {Printer p = new Printer( );// 开启一个线程new Thread(){@Overridepublic void run() {while(true){try {p.print1();} catch (InterruptedException e) {throw new RuntimeException(e);}}}}.start();
​// 又开启一个线程new Thread(){@Overridepublic void run() {while(true){try {p.print2();} catch (InterruptedException e) {throw new RuntimeException(e);}}}}.start();}
​
}

2、 同步代码块

再使用同步代码块演示一遍,再次确定一个结论

  • 锁对象是谁,就使用哪个对象来调用wait和notify

java">public class Printer2 {
​// 定义一个信号量// 1代表print1执行 2代表print2执行private int flag = 1;
​/*** 线程通信的要求* 1) 要保证线程安全* 2) 线程等待方法是wait*    线程唤醒方法是notify* 3) 必须使用锁对象调用 通信的方法* ---------* 为什么,wait和notify这些线程通信的方法要设计在Object类?* 答:*/public void print1() throws InterruptedException {synchronized (Object.class) {if (flag != 1) { // 信号不是1,说明不该print1执行,那就等待Object.class.wait( );}System.out.print(1 + " ");System.out.print(2 + " ");System.out.print(3 + " ");System.out.print(4 + " ");System.out.println( );
​// 改变信号量flag = 2;// 通知处于等待状态的线程启动Object.class.notify( );}}public void print2() throws InterruptedException {synchronized (Object.class) {if (flag != 2) { // 信号不是2,说明不该print2执行,那就等待Object.class.wait( );}System.out.print("A ");System.out.print("B ");System.out.print("C ");System.out.print("D ");System.out.println( );
​flag = 1;// 通知处于等待状态的线程启动Object.class.notify( );}}
}

补充: 目前这个代码可以保证两个线程通信,如果>= 3个线程,就不一定能按照预想顺序完成

原因是,线程过多,但是notify方法只能随机唤醒一个处于等待状态的线程

解决方案: 使用notifyAll

3、 wait和sleep

  • wait

    • 是Object类中的方法

    • wait会让线程等待

    • wait方法必须在同步方法中使用

    • wait方法方法线程等待时,会让出资源,别的线程可以执行

  • sleep

    • 是Thread类中的方法

    • sleep会让线程等待

    • 方法同步或者不同步都可以使用

      • 如果线程不安全,使用了sleep,会让出资源,别的线程执行

      • 如果线程安全,使用了sleep,不会释放资源,别的线程不会执行,会阻塞 --> 抱着锁睡


http://www.ppmy.cn/server/103652.html

相关文章

SAP Parallel Accounting(平行分类账业务)配置及操作手册【适用于多国家会计准则】

1. 配置准备 1.1 理解平行账概念 平行账,也称为多分类账,是SAP系统中的一项功能,它允许企业按照不同的会计准则来维护各自的财务数据。这种设置特别适用于那些需要符合多种会计准则的跨国公司。通过平行账,企业可以在不同的分类…

26-vector arraylist和linkedlist的区别

‌Vector, ArrayList, 和 LinkedList 是Java中常见的三种列表实现,它们各自具有不同的特点和适用场景。‌ ‌同步性与线程安全‌: ‌Vector‌ 是同步的,即线程安全的,它的所有方法都是同步的,可以由两个线程安全地访问…

通过相机来获取图片

文章目录 1. 概念介绍2. 方法与细节2.1 实现方法2.2 具体细节 3. 示例代码4. 内容总结 我们在上一章回中介绍了"如何混合选择多个图片和视频文件"相关的内容,本章回中将介绍如何通过相机获取图片文件.闲话休提,让我们一起Talk Flutter吧。 1. …

【赵渝强老师】Spark SQL的数据模型:DataFrame

通过SQL语句处理数据的前提是需要创建一张表,在Spark SQL中表被定义DataFrame,它由两部分组成:表结构的Schema和数据集合RDD,下图说明了DataFrame的组成。 从图中可以看出RDD是一个Java对象的数据集合,而DataFrame增加…

【精选】基于Python的热门旅游景点数据分析系统的设计与实现(南京旅游,北京旅游,旅游网站,全国各地旅游网站)

目录: 系统简介: 关键技术介绍 2.1 PYTHON语言简介 2.2 MySql数据库 2.3 DJANGO框架 2.4 Hadoop介绍 2.5 Scrapy介绍 2.6 B/S架构 系统总功能结构设计 系统详细实现: 6系统测试 系统测试的目的 软件测试过程 测试用例 为什么选择…

vue3+websocket的创建和使用,页面刷新保持长连接

最近一期的项目使用到了websocket,原因当然是因为长连接的特性,可以让网页实时接收到最新数据并更新页面。 区别于http,只能由用户端发送请求,服务端给出响应。 而websocket的特点在于可以主动给用户推送消息,因为二者…

【pdf文件生成】如何将盖章的文件生成PDF文件

一、提出问题 在我们的工作中,有时候上级让下级将盖章的文件生成PDF文件通过内部平台发送到上级邮箱,那如何解决呢?是去找一个扫描仪,还是用手机拍图转。用Python基实就能实现。 二、分析问题 现在网上好多的软件都是收费的&am…

数学建模预测类—【非线性回归】

每日格言:赞美那已经失去的,好让回忆变得可爱 前言 本篇我们将会从一般非线性回归求解和多项式回归求解两个方面来具体介绍如何进行建模求解 一、一般非线性回归求解 1、配曲线求解 适用条件:六类基本曲线:倒指数函数曲线&#…