前言:
等待/通知机制主要由Object类中的wait()、notify() 和 notifyAll()三个方法来实现,这三个方法均非Thread类中所声明的方法,而是Object类中声明的方法;原因是每个对象都拥有monitor(锁),所以让当前线程等待某个对象的锁,当然应该通过这个对象来操作,而不是用当前线程来操作
wait()——让当前线程释放对象锁并进入等待(阻塞)状态,在调用wait方法之前,线程必须获得该对象的对象级别锁,即只能在同步方法或同步块中调用wait方法。
notify()——唤醒一个正在等待相应对象锁的线程,使其进入就绪队列,以便在当前线程释放锁后竞争锁,进而得到CPU的执行,和wait方法一样,线程也必须获得该对象的对象级别锁,也只能在同步方法或同步块中调用。
notifyAll()——唤醒所有正在等待相应对象锁的所有线程,其它和notify一样。
一、举例说明
这里举一个生产面包和销售面包的例子,代码如下
1. 声明一个面包类
public class Bread {
public String name;
public int num;
Bread(String name, int num) {
System.out.println("面包名称:" + name + ", 初始化数量:" + num);
this.name = name;
this.num = num;
}
}
2. 声明一个生产面包的类,produce() 方法每次调用会生产一个面包,其调用受run()方法控制。
public class BreadProducer implements Runnable {
private final Bread bread;
private int maxNum;
BreadProducer(Bread bread, int maxNum) {
this.bread = bread;
this.maxNum = maxNum;
}
public void produce() {
bread.num = bread.num + 1;
System.out.println("面包生产1个,当前数量:" + bread.num);
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
public void run() {
synchronized (bread) {
if (bread.num >= maxNum) {
System.out.println("面包生产数量达到巅峰,停止生产,等待卖出");
try {
System.out.println("***************");
bread.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
} else {
System.out.println("---------------");
produce();
bread.notifyAll(); // 调用生产方法以后会唤醒wait等待的线程
}
}
}
}
3. 声明一个售卖面包的类,sale() 方法每次调用会销售一个面包,其调用受run()控制
public class BreadSale implements Runnable {
private final Bread bread;
BreadSale(Bread bread) {
this.bread = bread;
}
private void sale() {
bread.num = bread.num - 1;
System.out.println("面包售卖一个,当前剩余:" + bread.num);
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
public void run() {
synchronized (bread) {
if (bread.num <= 0) {
System.out.println("面包已卖完,等待生产");
try {
bread.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
} else {
sale();
bread.notifyAll(); // 调用销售方法以后会唤醒wait等待的线程
}
}
}
}
4. 写一个测试方法
package thread.wait;
public class Main {
public static void main(String[] args) throws InterruptedException {
Bread bread = new Bread("土司", 1);
BreadProducer breadProducer = new BreadProducer(bread, 2);
BreadSale breadSale = new BreadSale(bread);
Thread tp = new Thread(breadProducer);
Thread ts = new Thread(breadSale);
ts.start();
tp.start();
}
}
结果(两个线程启动顺序不同结果不同):
面包名称:土司, 初始化数量:1
面包售卖一个,当前剩余:0
---------------
面包生产1个,当前数量:1
根据上面的输出,可以判断:
1). 线程ts先启动执行,调用 BreadSale 中的run()方法,run()方法调用 sale() 方法,而后唤醒所有等待bread锁的进程(看结果唤醒的是 tp 进程),而后该线程逐步结束
2). 线程tp启动后调用 BreadProducer 中的run()方法,run()方法调用 produce() 方法,而后唤醒所有等待bread锁的进程(这时候tp已经结束,没有线程等待,所以直接结束了)
二、举例延申
考虑到上述是生产面包售卖面包是一个持续的变化过程(即通常描述的生产-消费模式),可以对两个多线程类稍加改造,加入while(true) 的循环,这样就能一直持续这种生产-销售过程。
public class BreadProducer implements Runnable {
private final Bread bread;
private int maxNum;
BreadProducer(Bread bread, int maxNum) {
this.bread = bread;
this.maxNum = maxNum;
}
public void produce() {
bread.num = bread.num + 1;
System.out.println("面包生产1个,当前数量:" + bread.num);
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
public void run() {
synchronized (bread) {
while (true) {
if (bread.num >= maxNum) {
System.out.println("面包生产数量达到巅峰,停止生产,等待卖出");
try {
System.out.println("***************");
bread.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
} else {
System.out.println("---------------");
produce();
bread.notifyAll();
}
}
}
}
}
public class BreadSale implements Runnable {
private final Bread bread;
BreadSale(Bread bread) {
this.bread = bread;
}
private void sale() {
bread.num = bread.num - 1;
System.out.println("面包售卖一个,当前剩余:" + bread.num);
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
public void run() {
synchronized (bread) {
while (true) {
if (bread.num <= 0) {
System.out.println("面包已卖完,等待生产");
try {
bread.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
} else {
sale();
bread.notifyAll();
}
}
}
}
}
public class Main {
public static void main(String[] args) throws InterruptedException {
Bread bread = new Bread("土司", 3);
BreadProducer breadProducer = new BreadProducer(bread, 5);
BreadSale breadSale = new BreadSale(bread);
Thread tp = new Thread(breadProducer);
Thread ts = new Thread(breadSale);
ts.start();
tp.start();
}
}
启动 main() 函数以后,可以看到一个持续生产-售卖的过程,永不停息(注意:慎用死循环)