当前位置:网站首页>Chapter 5: Multithreaded Communication—wait and notify
Chapter 5: Multithreaded Communication—wait and notify
2022-08-05 10:19:00 【Full stack programmer webmaster】
大家好,又见面了,我是你们的朋友全栈君.
线程通信概念:线程是操作系统中独立的个体,但这些个体如果不经过特殊的处理就不能成为一个整体,线程间的通信就成为整体的必用方式之一.当线程存在通信指挥,Interoperability between systems will be greater,在提高CPU利用率的同时还会使开发人员对线程任务在处理的过程中进行有效的把控与监督.
使用wait/notify方法实现线程间的通信.(注意这两个方法都是object的类的方法,换句话说java为所有的对象都提供了这两个方法)
1.wait和notify必须配合synchronized关键字使用
2.wait方法释放锁,notify方法不释放锁.
Let's take a look at an Alibaba interview question,Let's look at the class code shown below,代码的意思是在ListAdd1类中添加了一个add方法,该方法向list中添加字符串,size方法返回list的大小.线程”t1″调用10次add方法,每调用一次休眠0.5秒(这样线程”t2″There will be time to judgelist的大小),线程”t2″There is an endless loop of judgmentlistsize is reached5了,如果到5了,Then log and throw an exception to end the infinite loop.
package com.xiaoyexinxin.ThreadLearn;
import java.util.ArrayList;
import java.util.List;
public class ListAdd1 {
private volatile static List list = new ArrayList();
public void add(){
list.add("winner");
}
public int size(){
return list.size();
}
public static void main(String[] args){
final ListAdd1 list1 = new ListAdd1();
Thread t1 = new Thread(new Runnable() {
public void run() {
try {
for(int i=0;i<10;i++){
list1.add();
System.out.println("当前线程:"+Thread.currentThread().getName()+"添加了一个元素..");
Thread.sleep(500);
}
} catch (Exception e) {
e.printStackTrace();
}
}
},"t1");
Thread t2 = new Thread(new Runnable() {
public void run() {
while(true){
if(list.size() == 5){
System.out.println("The current thread is notified:"+Thread.currentThread().getName()+"list size=5线程停止..");
throw new RuntimeException();
}
}
}
},"t2");
t1.start();
t2.start();
}
}
我们来看执行结果,It can be seen that the results are consistent with what we designed,But this design is bad,Because it needs threads”t2″keep judginglist的大小,This is performance-intensive.
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
The current thread is notified:t2list size=5线程停止..
Exception in thread "t2" java.lang.RuntimeException
at com.xiaoyexinxin.ThreadLearn.ListAdd1$2.run(ListAdd1.java:39)
at java.lang.Thread.run(Unknown Source)
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
Then the first question is:请使用wait和notifyto improve the above code
使用wait和notify第一版代码如下,可以看到加上了synchronizedkeyword and usedwait和notify,特别需要注意的是,Threads are started in the order that they are started firstt2然后启动t1.
package com.xiaoyexinxin.ThreadLearn;
import java.util.ArrayList;
import java.util.List;
public class ListAdd1 {
private volatile static List list = new ArrayList();
public void add(){
list.add("winner");
}
public int size(){
return list.size();
}
public static void main(String[] args){
final ListAdd1 list2 = new ListAdd1();
//1.实例化出来一个lock
//当使用wait和notify的时候,一定要配合着synchronized关键字去使用
final Object lock = new Object();
Thread t1 = new Thread(new Runnable() {
public void run() {
try {
//线程t1和t2Be sure to use the same lock,Just use itlock
synchronized (lock) {
for(int i=0;i<10;i++){
list2.add();
System.out.println("当前线程:"+Thread.currentThread().getName()+"添加了一个元素..");
Thread.sleep(500);
if(list2.size() == 5){
System.out.println("Notice has been issued");
lock.notify(); //唤醒等待的锁,But also execute the current lock content first
}
}
}
} catch (Exception e) {
e.printStackTrace();
}
}
},"t1");
Thread t2 = new Thread(new Runnable() {
public void run() {
synchronized (lock) {
if(list2.size() != 5){
try {
lock.wait(); //线程等待,资源交给其他的线程
} catch (Exception e) {
e.printStackTrace();
}
}
System.out.println("当前线程"+Thread.currentThread().getName()+"收到通知,线程停止..");
throw new RuntimeException();
}
}
},"t2");
t2.start(); //让t2线程先执行
t1.start();
}
}
执行结果:
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
Notice has been issued
eee
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
当前线程t2收到通知,线程停止..
Exception in thread "t2" #######3333333333333333
java.lang.RuntimeException
at com.xiaoyexinxin.ThreadLearn.ListAdd1$2.run(ListAdd1.java:58)
at java.lang.Thread.run(Unknown Source)
This result is due to wait是释放锁的,而notify是不释放锁的,线程”t2″先执行,A judgment foundlist2.size不等于5,Hence the thread”t2″进入wait状态,释放了锁,这时线程”t1″The lock is acquired and execution begins,当线程”t1″添加5After each element is judged and foundlist2.size是5了,So it came out”已发出通知”的日志,lock.notify去唤醒”t2″线程,但是由于notify并不释放锁,因此线程”t1″Still holding the lock to execute the code behind,直到线程”t1″执行完后,线程”t2″The lock is obtained and the execution starts,So print it out”收到通知,线程停止”并抛出异常.
That is if you let the thread”t1″先执行,线程”t2″后执行,会是什么结果呢?We swap the startup order of the two threads,如下所示.
t1.start();
t2.start(); //让t2线程先执行
运行结果如下,发现线程”t1″执行完了,但是线程”t2″一直在等待,无法结束.
当前线程:t1添加了一个元素.. 当前线程:t1添加了一个元素.. 当前线程:t1添加了一个元素.. 当前线程:t1添加了一个元素.. 当前线程:t1添加了一个元素.. Notice has been issued eee 当前线程:t1添加了一个元素.. 当前线程:t1添加了一个元素.. 当前线程:t1添加了一个元素.. 当前线程:t1添加了一个元素.. 当前线程:t1添加了一个元素..
#######3333333333333333
Why does the above situation occur?This is due to the thread being executed first”t1″的话,线程”t1″The lock is obtained first to start execution,当list2The number of elements in is reached5While threading”t1″调用了lock.notify();但是由于notify并不释放锁,因此线程”t1″继续向下执行,list2Continue adding elements until the number of elements is reached10,线程”t1″结束,这时线程”t2″The lock is obtained and the execution starts,但由于list2.size这时已经是10了,never will be5了,因此线程”t2″判断list2The number of elements is not equal to 5,于是线程”t2″进入wait状态,线程”t1″已经结束了,There is no thread to wake up the thread”t2″了,因此线程”t2″It has been in a state of waiting.
我们可以看到,The current way of handling it(线程t2先执行,t1后执行)不好,因为线程”t2″要等到线程”t1″The notification can be received only after the execution is completed,This obviously does not meet the real-time requirements. In this case, please see the second question of this question:Since the above method is unreasonable,请用java.util.concurrentA tool under the package to achieve real-time receiving notifications.答案如下:The tool class we use is CountDownLatch,Another advantage of this class is that we don't have to write itsynchronized关键字修饰了,我们在线程”t2″Call the await method(countDownLatch.await();),在线程”t1″调用唤醒方法(countDownLatch.countDown();).And we don't have to struggle with threads either”t1″和”t2″The question of who starts first and who starts later,Whoever starts first can.
package com.xiaoyexinxin.ThreadLearn;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.CountDownLatch;
public class ListAdd3 {
private volatile static List list = new ArrayList();
public void add(){
list.add("winner");
}
public int size(){
return list.size();
}
public static void main(String[] args){
final ListAdd3 list2 = new ListAdd3();
//This is a very useful tool class under the concurrent package,实例化时的参数1Delegate needs to be called several times
//countDownLatch.countDown();to wake up,1就是调用1次即可,2to tune2次才行,
//我们一般都用1就行了
final CountDownLatch countDownLatch = new CountDownLatch(1);
Thread t1 = new Thread(new Runnable() {
public void run() {
try {
for(int i=0;i<10;i++){
list2.add();
System.out.println("当前线程:"+Thread.currentThread().getName()+"添加了一个元素..");
Thread.sleep(500);
if(list2.size() == 5){
System.out.println("Notice has been issued");
countDownLatch.countDown();
}
}
} catch (Exception e) {
e.printStackTrace();
}
}
},"t1");
Thread t2 = new Thread(new Runnable() {
public void run() {
if(list2.size() != 5){
try {
countDownLatch.await();
} catch (Exception e) {
e.printStackTrace();
}
}
System.out.println("当前线程"+Thread.currentThread().getName()+"收到通知,线程停止..");
throw new RuntimeException();
}
},"t2");
t1.start();
t2.start();
}
}
可以看到线程”t2″Received notifications in real time(我们不必纠结于”t2″Printed before the thread stops6条”t1″Add element information,It's a matter of the order of printing,We see that there are four pieces of information about adding elements in the back).It can be seen that this tool class is still very useful.
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
Notice has been issued
Exception in thread "t2" java.lang.RuntimeException
at com.xiaoyexinxin.ThreadLearn.ListAdd3$2.run(ListAdd3.java:54)
at java.lang.Thread.run(Unknown Source)
当前线程:t1添加了一个元素..
当前线程t2收到通知,线程停止..
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
当前线程:t1添加了一个元素..
发布者:全栈程序员栈长,转载请注明出处:https://javaforall.cn/106147.html原文链接:https://javaforall.cn
边栏推荐
- FPGA: Use of the development environment Vivado
- Jenkins manual (2) - software configuration
- A small test of basic grammar, Go lang1.18 introductory refining tutorial, from Bai Ding to Hongru, basic grammar of go lang and the use of variables EP02
- FPGA:基础入门按键控制LED灯
- MySQL data view
- Ali's new launch: Microservices Assault Manual, all operations are written out in PDF
- Can MySQL use aggregate functions without GROUP BY?
- 【Office】Microsoft Office下载地址合集(微软官方原版离线安装下载)
- [Unity] [UGUI] [Display text on the screen]
- STM32+ULN2003 drives 28BYJ4 stepper motor (forward and reverse according to the number of turns)
猜你喜欢
Meteorological data processing example - matlab string cutting matching and R language date matching (data splicing)
MySQL事务
[强网杯2022]WP-UM
用KUSTO查询语句(KQL)在Azure Data Explorer Database上查询LOG实战
Still looking for a network backup resources?Hurry up to collect the following network backup resource search artifact it is worth collecting!
【温度预警程序de开发】事件驱动模型实例运用
hcip BGP enhancement experiment
Ali's new launch: Microservices Assault Manual, all operations are written out in PDF
基于MindSpore高效完成图像分割,实现Dice!
RT-Thread记录(一、RT-Thread 版本、RT-Thread Studio开发环境 及 配合CubeMX开发快速上手)
随机推荐
Analysis and practice of antjian webshell dynamic encrypted connection
MySQL data view
【翻译】混沌网+SkyWalking:为混沌工程提供更好的可观察性
linux下oracle常见操作以及日常积累知识点(函数、定时任务)
012_SSS_ Improving Diffusion Model Efficiency Through Patching
Introduction to SD NAND Flash!
[强网杯2022]WP-UM
第四章:activiti RuntimeService设置获和取流程变量,及与taskService的区别,开始和完成任务时设置流程变量[通俗易懂]
用KUSTO查询语句(KQL)在Azure Data Explorer Database上查询LOG实战
导火索:OAuth 2.0四种授权登录方式必读
【Office】Microsoft Office下载地址合集(微软官方原版离线安装下载)
QSS 选择器
MySQL advanced (twenty-seven) database index principle
Jenkins manual (2) - software configuration
19.3 restart the Oracle environment
我们的Web3创业项目,黄了
第六章:activiti流程分流判断之排它网关和并行网关
百年北欧奢华家电品牌ASKO智能三温区酒柜臻献七夕,共品珍馐爱意
单片机:温度控制DS18B20
Why are RELTABLESPACE values 0 for many tables displayed in sys_class?