方法说明
1、wait()、notify/notifyAll() 方法是Object的本地final方法,无法被重写。
2、wait()执行后拥有当前锁的线程会释放该线程锁,并处于等待状态(等待重新获取锁)
3、notify/notifyAll() 执行后会唤醒处于等待状态线程获取线程锁、只是notify()只会随机唤醒其中之一获取线程锁,notifyAll() 会唤醒所有处于等待状态的线程抢夺线程锁。
三个方法都 必须 在synchronized 同步关键字所限定的作用域中调用,否则会报错java.lang.IllegalMonitorStateException ,意思是因为没有同步,所以线程对对象锁的状态是不确定的,不能调用这些方法。
- wait 表示持有对象锁的线程A准备释放对象锁权限,释放cpu资源并进入等待。
- notify 表示持有对象锁的线程A准备释放对象锁权限,通知jvm唤醒某个竞争该对象锁的线程X。线程A synchronized 代码作用域结束后,线程X直接获得对象锁权限,其他竞争线程继续等待(即使线程X同步完毕,释放对象锁,其他竞争线程仍然等待,直至有新的notify ,notifyAll被调用)。
- notifyAll 表示持有对象锁的线程A准备释放对象锁权限,通知jvm唤醒所有竞争该对象锁的线程,线程A synchronized 代码作用域结束后,jvm通过算法将对象锁权限指派给某个线程X,所有被唤醒的线程不再等待。线程X synchronized 代码作用域结束后,之前所有被唤醒的线程都有可能获得该对象锁权限,这个由JVM算法决定。
wait有三个重载方法,同时必须捕获非运行时异常InterruptedException。
- wait() 进入等待,需要notify ,notifyAll才能唤醒
- wait(long timeout) 进入等待,经过timeout 超时后,若未被唤醒,则自动唤醒
- wait(timeout, nanos) 进入等待,经过timeout 超时后,若未被唤醒,则自动唤醒。相对wait(long timeout) 更加精确时间。
示例
三个方法的最佳实践代码如下:
public class WaitAndNotify {
public static void main(String[] args) {
MethodClass methodClass = new MethodClass();
Thread t1 = new Thread(() -> {
try {
methodClass.product();
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}, "t1");
Thread t2 = new Thread(() -> {
try {
methodClass.customer();
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}, "t2");
Thread t3 = new Thread(() -> {
try {
methodClass.customer();
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}, "t3");
t1.start();
t2.start();
t3.start();
}
}
class MethodClass {
// 定义生产最大量
private final int MAX_COUNT = 20;
int productCount = 0;
public synchronized void product() throws InterruptedException {
while (true) {
System.out.println(Thread.currentThread().getName() + ":::run:::" + productCount);
Thread.sleep(10);
if (productCount >= MAX_COUNT) {
System.out.println("货舱已满,,.不必再生产");
wait();
} else {
productCount++;
}
notifyAll();
}
}
public synchronized void customer() throws InterruptedException {
while (true) {
System.out.println(Thread.currentThread().getName() + ":::run:::" + productCount);
Thread.sleep(10);
if (productCount <= 0) {
System.out.println("货舱已无货...无法消费");
wait();
} else {
productCount--;
}
notifyAll();
}
}
}