当前位置:网站首页>With 4 years of work experience, the 5 communication methods between multi-threads can't be said, can you believe it?
With 4 years of work experience, the 5 communication methods between multi-threads can't be said, can you believe it?
2022-08-03 22:11:00 【ah code】
问题
有两个线程,A 线程向一个集合里面依次添加元素“abc”字符串,一共添加十次,当添加到第五次的时候,希望 B 线程能够收到 A 线程的通知,然后 B 线程执行相关的业务操作.线程间通信的模型有两种:共享内存和消息传递,以下方式都是基本这两种模型来实现的.
一、使用 volatile 关键字
基于 volatile 关键字来实现线程间相互通信是使用共享内存的思想.大致意思就是多个线程同时监听一个变量,当这个变量发生变化的时候 ,线程能够感知并执行相应的业务.这也是最简单的一种实现方式
public class TestSync {//定义共享变量来实现通信,它需要volatile修饰,否则线程不能及时感知
static volatile boolean notice = false;
public static void main(String[] args) {
List<String> list = new ArrayList<>();
//线程A
Thread threadA = new Thread(() -> {
for ( int i = 1; i <= 10; i++) {
list.add( "abc");
System.out.println( "线程A添加元素,此时list的size为:" + list.size());
try {
Thread.sleep( 500);
} catch (InterruptedException e) {
e.printStackTrace();
}
if (list.size() == 5)
notice = true;
}
});
//线程B
Thread threadB = new Thread(() -> {
while ( true) {
if (notice) {
System.out.println( "线程B收到通知,开始执行自己的业务...");
break;
}
}
});
//需要先启动线程B
threadB.start();
try {
Thread.sleep( 1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
// 再启动线程A
threadA.start();
}
}
二、使用 Object 类的 wait()/notify()
Object 类提供了线程间通信的方法:wait()、notify()、notifyAll(),它们是多线程通信的基础,而这种实现方式的思想自然是线程间通信.
注意:wait/notify 必须配合 synchronized 使用,wait 方法释放锁,notify 方法不释放锁.wait 是指在一个已经进入了同步锁的线程内,让自己暂时让出同步锁,以便其他正在等待此锁的线程可以得到同步锁并运行,只有其他线程调用了notify(),notify并不释放锁,只是告诉调用过wait()的线程可以去参与获得锁的竞争了,但不是马上得到锁,因为锁还在别人手里,别人还没释放,调用 wait() 的一个或多个线程就会解除 wait 状态,重新参与竞争对象锁,程序如果可以再次得到锁,就可以继续向下运行.
public class TestSync {public static void main(String[] args) {
//定义一个锁对象
Object lock = new Object();
List<String> list = new ArrayList<>();
// 线程A
Thread threadA = new Thread(() -> {
synchronized (lock) {
for ( int i = 1; i <= 10; i++) {
list.add( "abc");
System.out.println( "线程A添加元素,此时list的size为:" + list.size());
try {
Thread.sleep( 500);
} catch (InterruptedException e) {
e.printStackTrace();
}
if (list.size() == 5)
lock.notify(); //唤醒B线程
}
}
});
//线程B
Thread threadB = new Thread(() -> {
while ( true) {
synchronized (lock) {
if (list.size() != 5) {
try {
lock.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
System.out.println( "线程B收到通知,开始执行自己的业务...");
}
}
});
//需要先启动线程B
threadB.start();
try {
Thread.sleep( 1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
//再启动线程A
threadA.start();
}
}
由输出结果,在线程 A 发出 notify() 唤醒通知之后,依然是走完了自己线程的业务之后,线程 B 才开始执行,正好说明 notify() 不释放锁,而 wait() 释放锁.
三、使用JUC工具类 CountDownLatch
jdk1.5 之后在java.util.concurrent包下提供了很多并发编程相关的工具类,简化了并发编程代码的书写,CountDownLatch 基于 ,相当于也是维护了一个线程间共享变量 state.
public class TestSync {public static void main(String[] args) {
CountDownLatch countDownLatch = new CountDownLatch( 1);
List<String> list = new ArrayList<>();
//线程A
Thread threadA = new Thread(() -> {
for ( int i = 1; i <= 10; i++) {
list.add( "abc");
System.out.println( "线程A添加元素,此时list的size为:" + list.size());
try {
Thread.sleep( 500);
} catch (InterruptedException e) {
e.printStackTrace();
}
if (list.size() == 5)
countDownLatch.countDown();
}
});
//线程B
Thread threadB = new Thread(() -> {
while ( true) {
if (list.size() != 5) {
try {
countDownLatch.await();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
System.out.println( "线程B收到通知,开始执行自己的业务...");
break;
}
});
//需要先启动线程B
threadB.start();
try {
Thread.sleep( 1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
//再启动线程A
threadA.start();
}
}
四、使用 ReentrantLock 结合 Condition
public class TestSync {public static void main(String[] args) {
ReentrantLock lock = new ReentrantLock();
Condition condition = lock.newCondition();
List<String> list = new ArrayList<>();
//线程A
Thread threadA = new Thread(() -> {
lock.lock();
for ( int i = 1; i <= 10; i++) {
list.add( "abc");
System.out.println( "线程A添加元素,此时list的size为:" + list.size());
try {
Thread.sleep( 500);
} catch (InterruptedException e) {
e.printStackTrace();
}
if (list.size() == 5)
condition.signal();
}
lock.unlock();
});
//线程B
Thread threadB = new Thread(() -> {
lock.lock();
if (list.size() != 5) {
try {
condition.await();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
System.out.println( "线程B收到通知,开始执行自己的业务...");
lock.unlock();
});
threadB.start();
try {
Thread.sleep( 1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
threadA.start();
}
}
这种方式使用起来并不是很好,代码编写复杂,而且线程 B 在被 A 唤醒之后由于没有获取锁还是不能立即执行,也就是说,A 在唤醒操作之后,并不释放锁.这种方法跟 Object 的 wait()/notify() 一样.
五、基本 LockSupport 实现线程间的阻塞和唤醒
LockSupport 是一种非常灵活的实现线程间阻塞和唤醒的工具,使用它不用关注是等待线程先进行还是唤醒线程先运行,但是得知道线程的名字.
public class TestSync {public static void main(String[] args) {
List<String> list = new ArrayList<>();
//线程B
final Thread threadB = new Thread(() -> {
if (list.size() != 5) {
LockSupport.park();
}
System.out.println( "线程B收到通知,开始执行自己的业务...");
});
//线程A
Thread threadA = new Thread(() -> {
for ( int i = 1; i <= 10; i++) {
list.add( "abc");
System.out.println( "线程A添加元素,此时list的size为:" + list.size());
try {
Thread.sleep( 500);
} catch (InterruptedException e) {
e.printStackTrace();
}
if (list.size() == 5)
LockSupport.unpark(threadB);
}
});
threadA.start();
threadB.start();
}
}
边栏推荐
- 基于支持向量机的网络⼊侵检测系统的全面调查和分类
- 目标检测的国内外研究现状
- Codeup刷题笔记-简单模拟
- 亿流量大考(2):开发一套高容错分布式系统
- 从0到1看支付
- 2022年全国职业院校技能大赛网络安全 B模块 任务十windows操作系统渗透测试 国赛原题
- [b01lers2020]Life on Mars
- 113. Teach a Man how to fish - How to query the documentation and technical implementation details of any SAP UI5 control property by yourself
- 【进阶自动化测试】一文1000教你如何用Postman做接口自动化测试
- [kali-vulnerability scanning] (2.1) Nessus lifts IP restrictions, scans quickly without results, and plugins are deleted (middle)
猜你喜欢

2022-08-02 mysql/stonedb慢SQL-Q18-内存使用暴涨分析

七夕快乐!

深度学习和机器学习有什么区别?

CAS:1797415-74-7_TAMRA-Azide-PEG-Biotin

encapsulation, package, access modifier, static variable

DO280管理和监控OpenShift平台--资源限制

投资性大于游戏性 NFT游戏到底是不是门好生意

CAS: 773888-45-2_BIOTIN ALKYNE_Biotin-alkynyl

数据一致性:双删为什么要延时?

【进阶自动化测试】一文1000教你如何用Postman做接口自动化测试
随机推荐
Makefile
三年黑盒测试工程师对嵌入式软件测试的理解
CAS:1620523-64-9_Azide-SS-biotin_biotin-disulfide-azide
Cross-end development technical reserve record
《强化学习周刊》第56期:GraphIRL、REDEEMER & 眼科强化学习的潜在研究
深度学习和机器学习有什么区别?
授人以渔 - 如何自行查询任意 SAP UI5 控件属性的文档和技术实现细节试读版
斩获双奖|易知微荣获“2021中国数字孪生解决方案优秀供应商”“中国智能制造优秀推荐产品”双奖项!
XSS testing
Makefile
Flink--Join以及Flink函数
目标检测的国内外研究现状
Nine ways to teach you to read the file path in the resources directory
472. Concatenated Words
noip初赛
Nacos配置文件管理、微服务获取Nacos配置文件
【MySQL进阶】数据库与表的创建和管理
Summary bug 】 【 Elipse garbled solution project code in Chinese!
LitJson报错记录
一文带你了解软件测试是干什么的?薪资高不高?0基础怎么学?