Java多线线程-----等待唤醒机制(wait notify)

目录

一.等待唤醒机制简介:

二.synchronized,wait(),notify():

三.等待唤醒机制案例:

例题一:

例题二:

四.什么时候释放锁—wait()、notify()


一.等待唤醒机制简介:

由于线程之间是抢占式执行的,因此线程的执行顺序难以预知。但是实际开发中有时候我们希望合理的协调多个线程之间的执行先后顺序。为了完成协调的工作,这里主要设计三个方法:

  • wait() / wait(long timeout) : 让当前线程进入等待状态

  • notify() / notifyAll(): 唤醒在当前对象上等待的线程

注意:wait,notify,notifyAll都是Object类的方法

二.synchronized,wait(),notify():

  • synchronized 的含义:

Java中每一个对象都可以成为一个监视器(Monitor), 该Monitor由一个锁(lock), 一个等待队列(waiting queue ), 一个入口队列( entry queue).

对于一个对象的方法, 如果没有synchronized关键字, 该方法可以被任意数量的线程,在任意时刻调用。

对于添加了synchronized关键字的方法,任意时刻只能被唯一的一个获得了对象实例锁的线程调用。

synchronized用于实现多线程的同步操作

  • wait()功能:

wait(), notify(), notifyAll() 和 synchonized 需要搭配使用, 用于线程同步

wait()总是在一个循;环中被调用,挂起当前线程来等待一个条件的成立。 Wait调用会一直等到其他线程调用notifyAll()时才返回。

当一个线程在执行synchronized 的方法内部,调用了wait()后, 该线程会释放该对象的锁, 然后该线程会被添加到该对象的等待队列中(waiting queue), 只要该线程在等待队列中, 就会一直处于闲置状态, 不会被调度执行。 要注意wait()方法会强迫线程先进行释放锁操作,所以在调用wait()时, 该线程必须已经获得锁,否则会抛出异常。由于wait()在synchonized的方法内部被执行, 锁一定已经获得, 就不会抛出异常了。

  • notify()的功能:

wait(), notify(), notifyAll() 和 synchonized 需要搭配使用, 用于线程同步

当一个线程调用一个对象的notify()方法时, 调度器会从所有处于该对象等待队列(waiting queue)的线程中取出任意一个线程, 将其添加到入口队列( entry queue) 中. 然后在入口队列中的多个线程就会竞争对象的锁, 得到锁的线程就可以继续执行。 如果等待队列中(waiting queue)没有线程, notify()方法不会产生任何作用

notifyAll() 和notify()工作机制一样, 区别在于notifyAll()会将等待队列(waiting queue)中所有的线程都添加到入口队列中(entry queue)

三.等待唤醒机制案例:

1.让t1执行wait()方法。

2.此时t2得到锁,再让t2执行notify()方法释放锁。

3.此时t1得到锁,t1会自动从wait()方法之后的代码,继续执行。

4.通过上述流程,我们就可以清楚的看到,wait()和notify()各自是怎么工作的了,也可以知道两者是怎么配合的了。

import java.util.*;
public class Main {//创建一个将被两个线程同时访问的共享对象public static Object loker = new Object();public static void main(String[] args) throws InterruptedException {Thread t1 = new Thread(()->{synchronized (loker){System.out.println("线程一初次获得对象锁,执行过程中调用锁对象的wait()方法~~");try {loker.wait();System.out.println("当线程一被唤醒后,后面的代码继续执行");} catch (InterruptedException e) {e.printStackTrace();}}System.out.println("线程一运行结束");},"线程一");Thread t2 = new Thread(()->{synchronized (loker){System.out.println("线程二初次获得对象锁,执行过程中调用锁对象的notify()方法~~");loker.notify();System.out.println("唤醒线程一前,后面的代码继续执行~~");}System.out.println("线程二结束");},"线程二");t1.start();//防止t2优先获得CPU执行权而错过唤醒t1Thread.sleep(1000);t2.start();}
}

运行结果(运行流程也就是运行的打印结果):

例题一:

有三个线程,线程名称分别为:a,b,c。每个线程打印自己的名称。

需要让他们同时启动,并按 c,b,a的顺序打印

代码详解:

import java.util.*;
public class Test {public static Object loker1 = new Object();public static Object loker2 = new Object();public static void main(String[] args) {System.out.println("打印顺序如下:");Thread t1 = new Thread(()->{//为了防止线程A的唤醒没有被线程B接受,这里先让线程A睡一会try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}System.out.println(Thread.currentThread().getName()+": C");synchronized (loker1){loker1.notify();}},"线程C");Thread t2 = new Thread(()->{synchronized (loker1){//等待线程A的唤醒try {loker1.wait();} catch (InterruptedException e) {e.printStackTrace();}}System.out.println(Thread.currentThread().getName()+": B");//线程B唤醒线程Csynchronized (loker2){loker2.notify();}},"线程B");Thread t3 = new Thread(()->{//线程C等待线程A的唤醒synchronized (loker2){try {loker2.wait();} catch (InterruptedException e) {e.printStackTrace();}}System.out.println(Thread.currentThread().getName()+": A");},"线程A");//开启线程t1.start();t2.start();t3.start();}
}

运行结果:

例题二:

有三个线程,分别只能打印A,B和C

要求按顺序打印ABC,打印10次

输出示例:

ABC

ABC

ABC

ABC

ABC

ABC

ABC

ABC

ABC

ABC

大致过程:

代码详解:

import java.util.*;
public class Demo {private static Object locker1 = new Object();private static Object locker2 = new Object();private static Object locker3 = new Object();public static void main(String[] args) throws InterruptedException {System.out.println("打印结果:");Thread t1 = new Thread(() -> {try {for (int i = 0; i < 10; i++) {synchronized (locker1) {locker1.wait();}System.out.print("A");synchronized (locker2) {locker2.notify();}}} catch (InterruptedException e) {e.printStackTrace();}});Thread t2 = new Thread(() -> {try {for (int i = 0; i < 10; i++) {synchronized (locker2) {locker2.wait();}System.out.print("B");synchronized (locker3) {locker3.notify();}}} catch (InterruptedException e) {e.printStackTrace();}});Thread t3 = new Thread(() -> {try {for (int i = 0; i < 10; i++) {synchronized (locker3) {locker3.wait();}System.out.println("C");synchronized (locker1) {locker1.notify();}}} catch (InterruptedException e) {e.printStackTrace();}});t1.start();t2.start();t3.start();//让三个线程都拿到锁Thread.sleep(1000);// 从线程 t1 启动synchronized (locker1) {locker1.notify();}}
}

运行结果:

四.什么时候释放锁—wait()、notify()

由于等待一个锁定线程只有在获得这把锁之后,才能恢复运行,所以让持有锁的线程在不需要锁的时候及时释放锁是很重要的。在以下情况下,持有锁的线程会释放锁:

1.执行完同步代码块。

2.在执行同步代码块的过程中,遇到异常而导致线程终止。

3.在执行同步代码块的过程中,执行了锁所属对象的wait()方法,这个线程会释放锁,进行对象的等待池。

除了以上情况外,只要持有锁的对象还没有执行完同步代码块,就不会释放锁。因此在以下情况下,线程不会释放锁:

1.在执行同步代码块的过程中,执行了Thread.sleep()方法,当前线程放弃CPU,开始睡眠,在睡眠中不会释放锁。

2.在执行同步代码块的过程中,执行了Thread.yield()方法,当前线程放弃CPU,但不会释放锁。

3.在执行同步代码块的过程中,其他线程执行了当前对象的suspend()方法,当前线程被暂停,但不会释放锁。但Thread类的suspend()方法已经被废弃。

结语: 写博客不仅仅是为了分享学习经历,同时这也有利于我巩固知识点,总结该知识点,由于作者水平有限,对文章有任何问题的还请指出,接受大家的批评,让我改进。同时也希望读者们不吝啬你们的点赞+收藏+关注,你们的鼓励是我创作的最大动力!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/50109.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

华为嵌入式面试题及参考答案(持续更新)

目录 详细讲TCP/IP协议的层数 材料硬度由什么决定? SD3.0接口电压标准 晶振市场失效率 RS232-C的硬件接口组成 详细讲眼图的功能 局域网传输介质有哪几类? 详细讲OSI模型 NMOS与PMOS的区别 I2C和SPI的区别 Static在C语言中的用法 堆栈和队列的区别 数组的时间复…

pyqt5制作音乐播放器(第三版)

这次接入了数据库&#xff0c;增加了翻页模式&#xff0c;更新了功能跳转之间的细节 数据设计&#xff1a; 收藏 like1时表示被收藏&#xff0c;展示show0的时候表示表数据被搜索 from peewee import Model, PrimaryKeyField, CharField, BooleanField, MySQLDatabase,Integer…

【区块链+绿色低碳】基于区块链的碳排放管理系统 | FISCO BCOS应用案例

目前业内的碳排放核查方式主要依靠于第三方人工核查、手动填报数据&#xff0c;然后由具备有认证资质的机构进行核验 盖章。但在此过程中存在数据造假的情况&#xff0c;给碳排放量核算的准确性、可靠性带来挑战。 中科易云采用国产开源联盟链 FISCO BCOS&#xff0c;推出基于…

搭建博客系统#Golang

WANLI 博客系统 项目介绍 基于vue3和gin框架开发的前后端分离个人博客系统&#xff0c;包含md格式的文本编辑展示&#xff0c;点赞评论收藏&#xff0c;新闻热点&#xff0c;匿名聊天室&#xff0c;文章搜索等功能。 项目已经部署并运行&#xff0c;快速开发可以查看博客&am…

GitHub每日最火火火项目(7.25)

1. 项目名称&#xff1a;public - apis / public - apis 项目介绍&#xff1a;这是一个集体列表&#xff0c;收集了各种免费的 APIs。在当今的软件开发中&#xff0c;API&#xff08;应用程序编程接口&#xff09;扮演着至关重要的角色&#xff0c;它们允许不同的应用程序和服…

基于Go语言开发调用高德API地址逆编码

最近公司有一个需求&#xff0c;有一批数据只有经纬度没有确定地址&#xff0c;现在需要根据经纬度补全地址&#xff0c;刚好高德提供这么一个API&#xff0c;可以拿来使用。 不过因为提供的数据的经纬度是大地2000坐标系&#xff0c;跟高德坐标系还不一样&#xff0c;需要进行…

培训第十一天(nfs与samba共享文件)

上午 1、环境准备 &#xff08;1&#xff09;yum源 &#xff08;一个云仓库pepl仓库&#xff09; [rootweb ~]# vim /etc/yum.repos.d/hh.repo [a]nameabaseurlfile:///mntgpgcheck0[rootweb ~]# vim /etc/fstab /dev/cdrom /mnt iso9660 defaults 0 0[rootweb ~]# mount -a[…

WebKit与PWA:打造无缝的渐进式Web应用体验

WebKit与PWA&#xff1a;打造无缝的渐进式Web应用体验 随着移动互联网的快速发展&#xff0c;用户对于Web应用的体验要求越来越高。Progressive Web Apps&#xff08;PWA&#xff09;&#xff0c;即渐进式Web应用&#xff0c;以其无需安装、易于更新、跨平台等特性&#xff0c…

JavaSE--基础语法--继承和多态(第三期)

一.继承 1.1我们为什么需要继承? 首先&#xff0c;Java中使用类对现实世界中实体来进行描述&#xff0c;类经过实例化之后的产物对象&#xff0c;则可以用来表示现实中的实体&#xff0c;但是 现实世界错综复杂&#xff0c;事物之间可能会存在一些关联&#xff0c;那在设计程…

Spring-一个接口拥有多实现类-企业应用场景

前言: 由于java的多态特性,往往一个接口有多种具体的实现,传统的做法是在一个实现类中新建不同方法。但这种做法既不符合OOP的思想,而且当每种实现逻辑都相对复杂的时候,会让我们的代码显得臃肿和凌乱,当我们只需要使用其中一种实现的时候,没有必要去关心其他实现,所以…

Java读取文件中多个JSON对象,并且16进制字符串和byte相互转换,将byte转为16进制字符串并写入json文件

Java读取文件中多个JSON对象 File file new File("/home/renjx/testcases.json");//将json转为mapsObjectMapper objectMapper new ObjectMapper();List<Map<String, String>> maps objectMapper.readValue(file, new TypeReference<List<Map&…

Apache虚拟主机VirtualHost配置项详解

在Apache中,VirtualHost容器用于定义一个虚拟主机的配置,它允许在单一的物理服务器上托管多个不同的网站,每个网站可以有自己的域名、文档根目录、错误日志等。VirtualHost内的配置项非常灵活,可以包含从基本的网站信息到高级的URL重写和安全设置。 以下是一些常见的Virtu…

Java之数组应用-冒泡排序-二分查找

冒泡排序 冒泡(Bubble Sort)排序是一种简单排序算法&#xff0c;它通过依次比较交换两个相邻元素实现功能。每一次冒泡会让至少一个元素移动到它应该在的位置上&#xff0c;这样 n 次冒泡就完成了 n 个数据的排序工作。 这个算法的名字由来是因为越小的元素会经由交换慢慢“浮”…

实在智能RPA助力三大运营商用“AI+RPA”打造新质生产力!

近年来&#xff0c;人工智能及自动化技术的突破性进展&#xff0c;正深刻地影响和重塑全球的生活生产模式。 作为我国现代化和数字化进程中的先行军的运营商行业&#xff0c;以中国电信、中国联通和中国移动等为代表的运营商企业&#xff0c;正致力于把握这一历史机遇&#xff…

SpringBoot项目配置多环境env

javaSpringBoot项目多环境配置 为什么maven Profiles 救命项目的pom文件管理 为什么 项目里面需要集成测试环境、开发、生产、多云环境&#xff0c;不仅需要application.yml,还需要加载别的config配置文件 故&#xff0c;我需要便捷多环境配置管理 maven Profiles 救命 项目的…

MySQL练手 --- 1934. 确认率

题目链接&#xff1a;1934. 确认率 思路 由题可知&#xff0c;两个表&#xff0c;一个表为Signups注册表&#xff0c;另一个表为Confirmations信息确认表&#xff0c;表的关联关系为 一对一&#xff0c;且user_id作为两个表的连接条件&#xff08;匹配字段&#xff09;&#…

【C# WInForm】将TextBox从输入框设置为文本框

1.需求情形&#xff1a; textbox作为最常用的控件之一&#xff0c;通常是用来输入文本信息或者显示文字&#xff0c;但是如果要在界面中显示大段文本&#xff0c;一个带有边框、可选中的文本样式似乎不合适。像这样&#xff1a; 我需要的是这段文字不仅能跨行&#xff0c;而且…

Atlassian Intelligence工具集解析:从自然语言到JQL处理,从虚拟代理到AI摘要、编辑器中的生成式AI等,全方位提升团队协作效率

2023年&#xff0c;Atlassian推出了Atlassian Intelligence——这是一款功能强大的团队协作增强工具&#xff0c;托管在其高级和企业级云平台上。Atlassian遵循 “释放团队潜力”的使命&#xff0c;利用合乎道德的AI模型来加速组织的现有能力。通过人机协作&#xff0c;用户可以…

判断文件格式

判断文件格式 判断文件格式,以便处理: if UpperCase(ExtractFileExt(sFileName)) <> .PDF thenbeginimgCapture.Picture.Graphic := nil;imgCapture.Picture.LoadFromFile(sFileName);end;unit SysUtils; function ExtractFileName(const FileName: string): string;/…

Unity3D 水面Mesh生成详解

在Unity3D中&#xff0c;创建逼真的水面效果是许多游戏和仿真项目中的重要部分。水面的动态效果通常通过Mesh&#xff08;网格&#xff09;的顶点动画来实现。本文将详细介绍如何在Unity3D中生成动态的水面Mesh&#xff0c;并给出具体的代码实现。. 对惹&#xff0c;这里有一个…