多线程(四)wait()、notify()以及notifyAll()
六、线程的等待和唤醒
1.wait()和notify()的简单示范
public class Wait extends Thread{
public synchronized void run()
{
System.out.println(getName()+"执行notify()");
notify();
}
public static void main(String []args) {
Wait w = new Wait();
synchronized (w){
try {
w.start();
System.out.println(Thread.currentThread().getName() + "睡眠");
Thread.sleep(3000);
System.out.println(Thread.currentThread().getName() + "等待");
w.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
程序结果:
main睡眠
main等待
Thread-0执行notify()
首先注意:wait()和notify()必须在synchronized代码块中。
这程序的流程是main线程先睡眠,然后被暂停,之后w线程执行notify()方法唤醒main(),如此就引出以下几个问题:
sleep()和wait()究竟有什么区别?
线程睡眠(sleep)不会放弃当前锁,而等待(wait)会。正如上面程序所示,即使线程睡眠的3000毫秒,但并没有放弃w的对象锁,所以run()方法不能获取w的对象锁,因此没有执行,体现在没有打印调用方法。而wait()执行后,main线程放弃对象锁,w线程获得自己的对象锁执行notify()方法,唤醒线程main,main继续执行。
w.wait()为什么是main线程暂停?
我们都知道sleep()是Thread类的静态方法,因此会睡眠执行该代码的线程。值得注意的是wait()虽然是Object()的实例方法,但是也具有sleep()类似的性质,即使运行代码的线程进入等待态。
notify()和notifyAll()的区别?
大部分人都耳熟能详,notify()唤醒监听器上一个线程,notifyAll()唤醒所有线程。notify()方法会随机唤醒一个线程,因此设计程序必须要合理。
wait()和wait(long mil)?
wait()需要notify()或者notifyAll()方法唤醒,而wait(long mil)同样,但是如果超过指定的时间还没有唤醒,那么自动唤醒。
这个程序其实是模仿博主如果天空不死的一个思路,而存在一个争论,即本程序中notify()方法删除,程序也会接着执行,原因位置,博主给出的解释是w线程结束后,释放的锁会由剩下的线程争抢,而该程序只剩下main线程,因此不必唤醒。显然有一点勉强,希望有大佬能解释。
2.wait()方法和notify()方法为何设置在Object类中?
wait()方法其实核心与sleep()不同就是放弃线程持有的对象锁,而notify()方法唤醒线程时只能在获取该对象锁的情况才有效(即唤醒线程和等待线程必须是同一把对象锁才能唤醒)。既然两者是通过对象锁联系到一起,那对象就有可能是几乎任何类的实例,因此将wait()、notify()和notifuAll()方法放在Object类中是合适的
3.一个简单的程序
我们通过wait()和notify()实现一个简单的功能,题目大致如下:
现有助理录不断入学生成绩,没录入一个学生,分析出总成绩和平均成绩
public class Grade {
int g;
static double avg=0;
static int num=0;
boolean b=true;
public synchronized void input(int g)
{
if(!b) {
try {
// System.out.println("暂停"+Thread.currentThread().getName());
wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
this.g = g;
System.out.println("写入成绩"+g);
b=!b;
notify();
}
public synchronized void analyze()
{
if(b) {
try {
// System.out.println("暂停"+Thread.currentThread().getName());
wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
double sum=(avg*num+g);
num++;
avg = sum / num;
System.out.println("Average of Grade is "+Grade.avg+", sum is "+sum);
b=!b;
notify();
}
public static void main(String []args)
{
Grade gr=new Grade();
C_Input c1=new C_Input(gr);
C_Analyze c2=new C_Analyze(gr);
c1.start();
c2.start();
}
}
class C_Input extends Thread
{
Grade grade;
public C_Input (Grade g)
{
grade=g;
}
public void run()
{
for(int i=0;i<5;i++) {
int mg=(int)(Math.random()*100);
grade.input(mg);
}
}
}
class C_Analyze extends Thread
{
Grade grade;
public C_Analyze (Grade g)
{
grade=g;
}
public void run()
{
for(int i=0;i<5;i++) {
grade.analyze();
}
}
}
结果:
写入成绩60
Average of Grade is 60.0, sum is 60.0
写入成绩84
Average of Grade is 72.0, sum is 144.0
写入成绩22
Average of Grade is 55.333333333333336, sum is 166.0
写入成绩96
Average of Grade is 65.5, sum is 262.0
写入成绩59
Average of Grade is 64.2, sum is 321.0
线程就先讲到这里,因为主要还是打基础,后面进阶我还会重新回来的!
多线程(四)wait()、notify()以及notifyAll()的更多相关文章
- Java多线程中wait, notify and notifyAll的使用
本文为翻译文章,原文地址:http://www.journaldev.com/1037/java-thread-wait-notify-and-notifyall-example 在Java的Obje ...
- 并发和多线程(四)--wait、notify、notifyAll、sleep、join、yield使用详解
wait.notify.notifyAll 这三个方法都是属于Object的,Java中的类默认继承Object,所以在任何方法中都可以直接调用wait(),notifyAll(),notify(), ...
- 多线程协作wait、notify、notifyAll方法简介理解使用 多线程中篇(十四)
在锁与监视器中有对wait和notify以及notifyAll进行了简单介绍 所有对象都有一个与之关联的锁与监视器 wait和notify以及notifyAll之所以是Object的方法就是因为任何一 ...
- Java多线程 wait, notify 和 notifyAll
Java的Object类 public class Object { public final native void notify(); public final native void notif ...
- Java多线程---------同步与死锁:synchronized;等待与唤醒:wait、notify、notifyAll;生命周期
1.问题的引出 class MyThread implements Runnable{ private int ticket = 5 ; // 假设一共有5张票 public void run(){ ...
- Java多线程的wait(),notify(),notifyAll()
在多线程的情况下.因为多个线程与存储空间共享相同的过程,同时带来的便利.它也带来了访问冲突这个严重的问题. Java语言提供了一种特殊的机制来解决这类冲突,避免同一数据对象由多个线程在同一时间访问. ...
- Java多线程:wait(),notify(),notifyAll()
1. wait(),notify(),notifyAll() 2. wait() 2.1. wait() 2.2. wait(long timeout) 2.3. wait(long timeout, ...
- java锁与监视器概念 为什么wait、notify、notifyAll定义在Object中 多线程中篇(九)
在Java中,与线程通信相关的几个方法,是定义在Object中的,大家都知道Object是Java中所有类的超类 在Java中,所有的类都是Object,借助于一个统一的形式Object,显然在有些处 ...
- Java线程和多线程(二)——对象中的wait,notify以及notifyAll方法
Java对象中的wait,notify以及notifyAll方法 在Java的Object类中包含了3个final的方法,这三个方法允许线程来交流资源是否被锁定.这三个方法就是wait(),notif ...
随机推荐
- PHP读取TXT中文乱码的解决方法
//$fname文件名称 if ($fname = $_FILES['nickname']['tmp_name']) { //file_get_contents() 函数把整个文件读入一个字符串中. ...
- visio给任意图形进行填充
使用visio给图形填充颜色的条件:图形必须是封闭的!!! 但是很多人不明白:我明明画的图形是封闭的啊!可是无法填充颜色!这是因为,你画了一个看上去是封闭的图形,但是VISIO认为你画的不是封闭图形! ...
- MYsql 客户端下载地址
workbench: https://www.mysql.com/cn/products/workbench/
- Spring4学习回顾之路02—IOC&DI
IOC&DI介绍 ●IOC:(Inversion of Control) :控制反转(反向获取资源) 其思想是反转资源获取的方向.传统的资源上查找方式要求组件向容器发起请求查找资源,作为回应, ...
- extern int PASCAL
表示声明一个变量,这个变量在其他地方已经定义,但是这里因为要使用,所以声明下. 写成下面: extern “C” int PASCAL: 说明PASCAL是在一个C文件下定义的.如果不是在C下就不用加 ...
- 怎样重置MySQL密码?
systemctl stop mysqld systemctl set-environment MYSQLD_OPTS="--skip-grant-tables" systemct ...
- pycharm2017.3.3永久激活(转载)
pycharm是很强大的开发工具,但是每次注册着实让人头疼.网络上很多注册码.注册服务器等等.但都只是一年或者不能用:为次有如下解决方案.亲测有效!!! 如果想让pycharm永久被激活,比如截止日到 ...
- 【题解】P3391 文艺平衡树
用pb_ds库中的rope水过去的,忽然发现这玩意能水好多模拟题. 详见这个博客:背景的小姐姐真的好看 声明 #include <ext/rope> using namespace __g ...
- (三)创建基于maven的javaFX+springboot项目创建
创建基于maven的javaFx+springboot项目有两种方式,第一种为通过非编码的方式来设计UI集成springboot:第二种为分离用户界面(UI)和后端逻辑集成springboot,其中用 ...
- init是一个自定义方法名
init是一个自定义方法名,用于初始化页面变量.上面的代码表示初始化方法是在当前网页加载后执行的(当浏览器打开网页时,触发窗口对象的onload方法,用上面的代码执行名为init的初始化方法).事实上 ...