synchronized wait()/notify 对比 ReentrantLock await()/signal()

2024-09-04 14:52

本文主要是介绍synchronized wait()/notify 对比 ReentrantLock await()/signal(),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

结论

synchronized

synchronized 配合 wait()/notify 无法实现精准唤醒线程

ReentrantLock

ReentrantLock 配合 Condition await()/signal() 可以实现精准唤醒线程 (指唤醒指定的线程)

ReentrantLock 如何实现精准唤醒线程

一个 lock 配合多个 Condition, 且每个 Condition 中只有一个线程
(若一个Condition中有多个线程,也无法精准唤醒线程)

案例 synchronized


public class ManySync {public void await1() {synchronized (this) {try {System.out.println("await1 - 进入等待" + Thread.currentThread().getName());this.wait();System.out.println("await1 - 被唤醒" + Thread.currentThread().getName());} catch (Exception e) {e.printStackTrace();}}}public synchronized void await2() {synchronized (this) {try {System.out.println("await2 - 进入等待" + Thread.currentThread().getName());this.wait();System.out.println("await2 - 被唤醒" + Thread.currentThread().getName());} catch (Exception e) {e.printStackTrace();}}}public void await3() {synchronized (this) {try {System.out.println("await3 - 进入等待" + Thread.currentThread().getName());this.wait();System.out.println("await3 - 被唤醒" + Thread.currentThread().getName());} catch (Exception e) {e.printStackTrace();}}}public void await4() {synchronized (this) {try {System.out.println("await4 - 进入等待" + Thread.currentThread().getName());this.wait();System.out.println("await4 - 被唤醒" + Thread.currentThread().getName());} catch (Exception e) {e.printStackTrace();}}}public  void notif() {synchronized(this) {try {this.notify();this.notify();this.notify();this.notify();} catch (Exception e) {e.printStackTrace();}}}}class Run3 {public static void main(String[] args) {try {ManySync manySync = new ManySync();Thread t1 = new Thread(() -> {manySync.await1();});Thread t2 = new Thread(() -> {manySync.await2();});Thread t3 = new Thread(() -> {manySync.await3();});Thread t4 = new Thread(() -> {manySync.await4();});t1.start();Thread.sleep(200);t2.start();Thread.sleep(200);t3.start();Thread.sleep(200);t4.start();Thread.sleep(1000);manySync.notif();} catch (Exception e) {e.printStackTrace();}}
}

输出结果 可以看出等待时间长的锁并不会最先被唤醒, 也无法唤醒指定线程,(因为并未提供能唤醒的API,也无法进行设计) 唤醒的输出结果是随机的 由 JVM 调度

await1 - 进入等待Thread-0
await2 - 进入等待Thread-1
await3 - 进入等待Thread-2
await4 - 进入等待Thread-3
await1 - 被唤醒Thread-0
await4 - 被唤醒Thread-3
await3 - 被唤醒Thread-2
await2 - 被唤醒Thread-1

案例 ReentrantLock


import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;public class ManyCondition {private Lock lock = new ReentrantLock();Condition condition1 = lock.newCondition();Condition condition2 = lock.newCondition();Condition condition3 = lock.newCondition();Condition condition4 = lock.newCondition();public void await1() {try {lock.lock();System.out.println("await1 - 进入等待" + Thread.currentThread().getName());condition1.await();System.out.println("await1 - 被唤醒" + Thread.currentThread().getName());lock.unlock();} catch (Exception e) {e.printStackTrace();}}public void await2() {try {lock.lock();System.out.println("await2 - 进入等待" + Thread.currentThread().getName());condition2.await();System.out.println("await2 - 被唤醒" + Thread.currentThread().getName());lock.unlock();} catch (Exception e) {e.printStackTrace();}}public void await3() {try {lock.lock();System.out.println("await3 - 进入等待" + Thread.currentThread().getName());condition3.await();System.out.println("await3 - 被唤醒" + Thread.currentThread().getName());lock.unlock();} catch (Exception e) {e.printStackTrace();}}public void await4() {try {lock.lock();System.out.println("await4 - 进入等待" + Thread.currentThread().getName());condition4.await();System.out.println("await4 - 被唤醒" + Thread.currentThread().getName());lock.unlock();} catch (Exception e) {e.printStackTrace();}}public void signal() {try {lock.lock();System.out.println("await1 - 被唤醒 - " + Thread.currentThread().getName());condition1.signal();lock.unlock();lock.lock();System.out.println("await2 - 被唤醒 - " + Thread.currentThread().getName());condition2.signal();lock.unlock();lock.lock();System.out.println("await3 - 被唤醒 - " + Thread.currentThread().getName());condition3.signal();lock.unlock();lock.lock();System.out.println("await4 - 被唤醒 - " + Thread.currentThread().getName());condition4.signal();lock.unlock();} catch (Exception e) {e.printStackTrace();}}}class Run2 {public static void main(String[] args) {try {ManyCondition manyCondition = new ManyCondition();Thread t1 = new Thread(() -> {manyCondition.await1();});Thread t2 = new Thread(() -> {manyCondition.await2();});Thread t3 = new Thread(() -> {manyCondition.await3();});Thread t4 = new Thread(() -> {manyCondition.await4();});t1.start();t2.start();t3.start();t4.start();Thread.sleep(2000);manyCondition.signal();} catch (Exception e) {e.printStackTrace();}}
}

输出结果 这里可以看出 ,当每个Condition 中只有一个线程时, 可以通过逻辑控制, 实现精准唤醒需要唤醒的线程

await1 - 进入等待Thread-0
await2 - 进入等待Thread-1
await3 - 进入等待Thread-2
await4 - 进入等待Thread-3
await1 - 被唤醒 - main
await2 - 被唤醒 - main
await3 - 被唤醒 - main
await4 - 被唤醒 - main
await1 - 被唤醒Thread-0
await2 - 被唤醒Thread-1
await3 - 被唤醒Thread-2
await4 - 被唤醒Thread-3

在这里插入图片描述

这篇关于synchronized wait()/notify 对比 ReentrantLock await()/signal()的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1136284

相关文章

Python实现Microsoft Office自动化的几种方式及对比详解

《Python实现MicrosoftOffice自动化的几种方式及对比详解》办公自动化是指利用现代化设备和技术,代替办公人员的部分手动或重复性业务活动,优质而高效地处理办公事务,实现对信息的高效利用... 目录一、基于COM接口的自动化(pywin32)二、独立文件操作库1. Word处理(python-d

Java常用注解扩展对比举例详解

《Java常用注解扩展对比举例详解》:本文主要介绍Java常用注解扩展对比的相关资料,提供了丰富的代码示例,并总结了最佳实践建议,帮助开发者更好地理解和应用这些注解,需要的朋友可以参考下... 目录一、@Controller 与 @RestController 对比二、使用 @Data 与 不使用 @Dat

python中字符串拼接的几种方法及优缺点对比详解

《python中字符串拼接的几种方法及优缺点对比详解》在Python中,字符串拼接是常见的操作,Python提供了多种方法来拼接字符串,每种方法有其优缺点和适用场景,以下是几种常见的字符串拼接方法,需... 目录1. 使用 + 运算符示例:优缺点:2. 使用&nbsjsp;join() 方法示例:优缺点:3

C++ 各种map特点对比分析

《C++各种map特点对比分析》文章比较了C++中不同类型的map(如std::map,std::unordered_map,std::multimap,std::unordered_multima... 目录特点比较C++ 示例代码 ​​​​​​代码解释特点比较1. std::map底层实现:基于红黑

Golang中拼接字符串的6种方式性能对比

《Golang中拼接字符串的6种方式性能对比》golang的string类型是不可修改的,对于拼接字符串来说,本质上还是创建一个新的对象将数据放进去,主要有6种拼接方式,下面小编就来为大家详细讲讲吧... 目录拼接方式介绍性能对比测试代码测试结果源码分析golang的string类型是不可修改的,对于拼接字

MySQL表锁、页面锁和行锁的作用及其优缺点对比分析

《MySQL表锁、页面锁和行锁的作用及其优缺点对比分析》MySQL中的表锁、页面锁和行锁各有特点,适用于不同的场景,表锁锁定整个表,适用于批量操作和MyISAM存储引擎,页面锁锁定数据页,适用于旧版本... 目录1. 表锁(Table Lock)2. 页面锁(Page Lock)3. 行锁(Row Lock

Python使用Pandas对比两列数据取最大值的五种方法

《Python使用Pandas对比两列数据取最大值的五种方法》本文主要介绍使用Pandas对比两列数据取最大值的五种方法,包括使用max方法、apply方法结合lambda函数、函数、clip方法、w... 目录引言一、使用max方法二、使用apply方法结合lambda函数三、使用np.maximum函数

一文详解Java Condition的await和signal等待通知机制

《一文详解JavaCondition的await和signal等待通知机制》这篇文章主要为大家详细介绍了JavaCondition的await和signal等待通知机制的相关知识,文中的示例代码讲... 目录1. Condition的核心方法2. 使用场景与优势3. 使用流程与规范基本模板生产者-消费者示例

锐捷和腾达哪个好? 两个品牌路由器对比分析

《锐捷和腾达哪个好?两个品牌路由器对比分析》在选择路由器时,Tenda和锐捷都是备受关注的品牌,各自有独特的产品特点和市场定位,选择哪个品牌的路由器更合适,实际上取决于你的具体需求和使用场景,我们从... 在选购路由器时,锐捷和腾达都是市场上备受关注的品牌,但它们的定位和特点却有所不同。锐捷更偏向企业级和专

什么是 Ubuntu LTS?Ubuntu LTS和普通版本区别对比

《什么是UbuntuLTS?UbuntuLTS和普通版本区别对比》UbuntuLTS是Ubuntu操作系统的一个特殊版本,旨在提供更长时间的支持和稳定性,与常规的Ubuntu版本相比,LTS版... 如果你正打算安装 Ubuntu 系统,可能会被「LTS 版本」和「普通版本」给搞得一头雾水吧?尤其是对于刚入