Java学习day26:和线程相关的Object类的方法、等待线程和唤醒线程(知识点详解)

本文主要是介绍Java学习day26:和线程相关的Object类的方法、等待线程和唤醒线程(知识点详解),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

声明:该专栏本人重新过一遍java知识点时候的笔记汇总,主要是每天的知识点+题解,算是让自己巩固复习,也希望能给初学的朋友们一点帮助,大佬们不喜勿喷(抱拳了老铁!)


往期回顾

Java学习day25:守护线程、死锁、线程生命周期(知识点详解)-CSDN博客

Java学习day24:线程的同步和锁(例题+知识点详解)-CSDN博客

Java学习day23:线程构造方法、常用方法(例题+知识点详解)-CSDN博客

 Java学习day26:和线程相关的Object类下面的方法、等待线程和唤醒线程

一、和线程相关的Object类的方法

1.三个常用方法

void wait()  导致当前线程等待,直到另一个线程调用该对象的 notify()方法或 notifyAll()方法。
void notify()  唤醒正在等待对象监视器的单个线程。
void notifyAll() 唤醒正在等待对象监视器的所有线程。

wait()  方法换句话说,这个方法的行为就好像简单地执行呼叫`wait(0)`

二、等待线程和唤醒线程

1.什么是等待线程和唤醒线程

上面我们看了,Object类的三个主要方法,这三个主要方法就涉及到等待线程和唤醒线程,也就是说,至少两个线程,其中一个线程中使用对象.wait()  那么这个线程就会阻塞,代码不会往下执行了。也就是我们说的,等待线程。如何想让这个线程往下执行呢?再开另外一个线程,使用对象.notify()去唤醒另外那个等待线程。也就是我们说的唤醒线程,如果多个等待线程,一个唤醒线程,则调用notifyAll()方法,就能一次性唤醒所有等待线程。

我们用代码来理解

示例:

//创建这个类的目的,就是实例化出来对象,然后拿这个对象
//调用wait方法和notify方法
//wait方法和notify方法是object对象的方法
class Message {private String message;public Message(String message) {this.message = message;}public String getMessage() {return message;}public void setMessage(String message) {this.message = message;}@Overridepublic String toString() {return "Message{" +"message='" + message + '\'' +'}';}}
//导致当前线程等待,直到另一个线程调用该对象的[`notify()`
class WaiterThread implements Runnable {//想一个问题?WaiterThread  使用message对象调用//wait() 咋解决?private Message msg;public WaiterThread(Message msg) {this.msg = msg;}@Overridepublic void run() {//先获取当前线程名字String name = Thread.currentThread().getName();System.out.println(name + "等待唤醒时间:" +System.currentTimeMillis());//让等待线程去阻塞,去等待  这个线程执行不下去了//锁的是msg对象synchronized (msg) {//为啥用这个wait的时候要加锁?等会讲try {msg.wait();//代码走到这,当前这个线程阻塞,不往下走了//咱们得想办法让这个等待线程继续执行下去,咋办?//在另外一个线程中去调用notify方法那么等待线程就会执行下去} catch (InterruptedException e) {e.printStackTrace();}System.out.println("123");System.out.println(name + "被唤醒的时间:" + System.currentTimeMillis());}}
}
//唤醒线程
class NotifyThread implements Runnable {//也要用同一个对象是WaiterThread线程中同一个对象调用notify()方法private Message msg;public NotifyThread(Message msg) {this.msg = msg;}@Overridepublic void run() {try {//我的想法是不能先让唤醒线程执行Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}String name = Thread.currentThread().getName();System.out.println(name + "开始唤醒等待线程");synchronized (msg) {msg.setMessage("我是修改之后的message值");msg.notify();//msg.notifyAll();//唤醒所有线程}}
}
public class Demo1 {public static void main(String[] args) {Message message = new Message("我是message属性");WaiterThread waiterThread = new WaiterThread(message);NotifyThread notifyThread = new NotifyThread(message);//如果等待线程好几个 咋办呢?new Thread(waiterThread, "等待线程1").start();// new Thread(waiterThread, "等待线程2").start();//new Thread(waiterThread, "等待线程3").start();new Thread(notifyThread, "唤醒线程").start();   }
}

代码执行结果:

        //等待线程等待唤醒时间:1660187660718   等待线程
        //唤醒线程开始唤醒等待线程        唤醒线程
        //123  等待线程
        //等待线程被唤醒的时间:1660187661740  等待线程
        //这叫线程之间的通信问题!!! 

理解这段代码核心:
先是写一个message方法用于信息显示,这个没啥好说的,然后写了两个类用来创建两个线程,分别是等待线程和唤醒线程,等待线程执行wait方法后就相当于不再执行接下来的代码,进入就绪等待状态,让唤醒线程获得cpu使用权,唤醒线程调用notify()方法后,等待线程再继续执行。同时代码里已经写清楚了,如果有多个等待线程,则唤醒线程的对象需要调用notifyAll()方法,一次性唤醒所有等待线程,而且我们必须明确的是,哪个等待线程先被唤醒是不知道的,因为抢占式运行,哪个线程先抢到cpu执行权,唤醒线程就先唤醒哪个线程。

总结:

新建两个线程:
 一个是等待线程
  线程里面的代码从上往下执行的,但是使用object.wait(),就这个方法一用,你的线程就
  阻塞了,就处于等待状态。意味着当前的代码到了wait方法以后的代码暂时不执行了
 另外一个是唤醒线程。
  唤醒线程中使用object.notify()方法,这个方法是专门唤醒刚才那个等待线程。让等待线程继续执行

同时,有一个很重要的点:调用wait()的等待线程必须加锁,具体为什么? 

wait需要持有锁的原因是,你肯定需要知道在哪个对象上进行等待,如果不持有锁,将无法做到对象变更时进行实时感知通知的作用。与此同时,为了让其他线程可以操作该值的变化,它必须要先释放掉锁,然后在该节点上进行等待。不持有锁而进行wait,可能会导致长眠不起。而且,如果不持有锁,则当wait之后的操作,都可能是错的,因为可能这个数据已经过时,其实也叫线程不安全了。总之,一切为了安全,单独的wait做不成这事。

通俗来说就是,因为我要知道对哪个对象进行唤醒的!!!
举个生活中的例子:
 大学的时候在楼底下等过自己的女朋友。
 你就是等待线程
 你女朋友就是唤醒线程
 你在楼底下等的时候  就是在wait。
 你女朋友下楼之后,唤醒你  咱们走吧。
 但是你女朋友 去唤醒别人的男朋友?这就扯犊子了,所有的加锁是为了保证是同一个对象的 

所以说,我们必须要对等待线程和唤醒线程都加锁。 


以上,就是今天的所有知识点了。等待线程和唤醒线程的代码大家要多花点时间理解,方法其实就三个,但是大家要知道怎么用的,有些代码细节都没讲到,大家要自己多花点时间,静下心看代码,写代码,多理解,多运用,重点是多去运用。

加油吧,预祝大家变得更强!

这篇关于Java学习day26:和线程相关的Object类的方法、等待线程和唤醒线程(知识点详解)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

C语言函数递归实际应用举例详解

《C语言函数递归实际应用举例详解》程序调用自身的编程技巧称为递归,递归做为一种算法在程序设计语言中广泛应用,:本文主要介绍C语言函数递归实际应用举例的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录前言一、递归的概念与思想二、递归的限制条件 三、递归的实际应用举例(一)求 n 的阶乘(二)顺序打印

idea maven编译报错Java heap space的解决方法

《ideamaven编译报错Javaheapspace的解决方法》这篇文章主要为大家详细介绍了ideamaven编译报错Javaheapspace的相关解决方法,文中的示例代码讲解详细,感兴趣的... 目录1.增加 Maven 编译的堆内存2. 增加 IntelliJ IDEA 的堆内存3. 优化 Mave

Java String字符串的常用使用方法

《JavaString字符串的常用使用方法》String是JDK提供的一个类,是引用类型,并不是基本的数据类型,String用于字符串操作,在之前学习c语言的时候,对于一些字符串,会初始化字符数组表... 目录一、什么是String二、如何定义一个String1. 用双引号定义2. 通过构造函数定义三、St

Python Faker库基本用法详解

《PythonFaker库基本用法详解》Faker是一个非常强大的库,适用于生成各种类型的伪随机数据,可以帮助开发者在测试、数据生成、或其他需要随机数据的场景中提高效率,本文给大家介绍PythonF... 目录安装基本用法主要功能示例代码语言和地区生成多条假数据自定义字段小结Faker 是一个 python

springboot filter实现请求响应全链路拦截

《springbootfilter实现请求响应全链路拦截》这篇文章主要为大家详细介绍了SpringBoot如何结合Filter同时拦截请求和响应,从而实现​​日志采集自动化,感兴趣的小伙伴可以跟随小... 目录一、为什么你需要这个过滤器?​​​二、核心实现:一个Filter搞定双向数据流​​​​三、完整代码

SpringBoot利用@Validated注解优雅实现参数校验

《SpringBoot利用@Validated注解优雅实现参数校验》在开发Web应用时,用户输入的合法性校验是保障系统稳定性的基础,​SpringBoot的@Validated注解提供了一种更优雅的解... 目录​一、为什么需要参数校验二、Validated 的核心用法​1. 基础校验2. php分组校验3

Java Predicate接口定义详解

《JavaPredicate接口定义详解》Predicate是Java中的一个函数式接口,它代表一个判断逻辑,接收一个输入参数,返回一个布尔值,:本文主要介绍JavaPredicate接口的定义... 目录Java Predicate接口Java lamda表达式 Predicate<T>、BiFuncti

详解如何通过Python批量转换图片为PDF

《详解如何通过Python批量转换图片为PDF》:本文主要介绍如何基于Python+Tkinter开发的图片批量转PDF工具,可以支持批量添加图片,拖拽等操作,感兴趣的小伙伴可以参考一下... 目录1. 概述2. 功能亮点2.1 主要功能2.2 界面设计3. 使用指南3.1 运行环境3.2 使用步骤4. 核

Spring Security基于数据库的ABAC属性权限模型实战开发教程

《SpringSecurity基于数据库的ABAC属性权限模型实战开发教程》:本文主要介绍SpringSecurity基于数据库的ABAC属性权限模型实战开发教程,本文给大家介绍的非常详细,对大... 目录1. 前言2. 权限决策依据RBACABAC综合对比3. 数据库表结构说明4. 实战开始5. MyBA

Spring Security方法级安全控制@PreAuthorize注解的灵活运用小结

《SpringSecurity方法级安全控制@PreAuthorize注解的灵活运用小结》本文将带着大家讲解@PreAuthorize注解的核心原理、SpEL表达式机制,并通过的示例代码演示如... 目录1. 前言2. @PreAuthorize 注解简介3. @PreAuthorize 核心原理解析拦截与