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实现Word文档的自动化对比方案

《使用Python实现Word文档的自动化对比方案》我们经常需要比较两个Word文档的版本差异,无论是合同修订、论文修改还是代码文档更新,人工比对不仅效率低下,还容易遗漏关键改动,下面通过一个实际案例... 目录引言一、使用python-docx库解析文档结构二、使用difflib进行差异比对三、高级对比方

Python异步编程之await与asyncio基本用法详解

《Python异步编程之await与asyncio基本用法详解》在Python中,await和asyncio是异步编程的核心工具,用于高效处理I/O密集型任务(如网络请求、文件读写、数据库操作等),接... 目录一、核心概念二、使用场景三、基本用法1. 定义协程2. 运行协程3. 并发执行多个任务四、关键

Java实现本地缓存的四种方法实现与对比

《Java实现本地缓存的四种方法实现与对比》本地缓存的优点就是速度非常快,没有网络消耗,本地缓存比如caffine,guavacache这些都是比较常用的,下面我们来看看这四种缓存的具体实现吧... 目录1、HashMap2、Guava Cache3、Caffeine4、Encache本地缓存比如 caff

MySQL中读写分离方案对比分析与选型建议

《MySQL中读写分离方案对比分析与选型建议》MySQL读写分离是提升数据库可用性和性能的常见手段,本文将围绕现实生产环境中常见的几种读写分离模式进行系统对比,希望对大家有所帮助... 目录一、问题背景介绍二、多种解决方案对比2.1 原生mysql主从复制2.2 Proxy层中间件:ProxySQL2.3

C# async await 异步编程实现机制详解

《C#asyncawait异步编程实现机制详解》async/await是C#5.0引入的语法糖,它基于**状态机(StateMachine)**模式实现,将异步方法转换为编译器生成的状态机类,本... 目录一、async/await 异步编程实现机制1.1 核心概念1.2 编译器转换过程1.3 关键组件解析

MySQL中EXISTS与IN用法使用与对比分析

《MySQL中EXISTS与IN用法使用与对比分析》在MySQL中,EXISTS和IN都用于子查询中根据另一个查询的结果来过滤主查询的记录,本文将基于工作原理、效率和应用场景进行全面对比... 目录一、基本用法详解1. IN 运算符2. EXISTS 运算符二、EXISTS 与 IN 的选择策略三、性能对比

详解MySQL中JSON数据类型用法及与传统JSON字符串对比

《详解MySQL中JSON数据类型用法及与传统JSON字符串对比》MySQL从5.7版本开始引入了JSON数据类型,专门用于存储JSON格式的数据,本文将为大家简单介绍一下MySQL中JSON数据类型... 目录前言基本用法jsON数据类型 vs 传统JSON字符串1. 存储方式2. 查询方式对比3. 索引

SpringBoot中六种批量更新Mysql的方式效率对比分析

《SpringBoot中六种批量更新Mysql的方式效率对比分析》文章比较了MySQL大数据量批量更新的多种方法,指出REPLACEINTO和ONDUPLICATEKEY效率最高但存在数据风险,MyB... 目录效率比较测试结构数据库初始化测试数据批量修改方案第一种 for第二种 case when第三种

关于MyISAM和InnoDB对比分析

《关于MyISAM和InnoDB对比分析》:本文主要介绍关于MyISAM和InnoDB对比分析,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录开篇:从交通规则看存储引擎选择理解存储引擎的基本概念技术原理对比1. 事务支持:ACID的守护者2. 锁机制:并发控制的艺

CSS中的Static、Relative、Absolute、Fixed、Sticky的应用与详细对比

《CSS中的Static、Relative、Absolute、Fixed、Sticky的应用与详细对比》CSS中的position属性用于控制元素的定位方式,不同的定位方式会影响元素在页面中的布... css 中的 position 属性用于控制元素的定位方式,不同的定位方式会影响元素在页面中的布局和层叠关