概述
被唤醒之后,一些数据已经发生了变化,需要对数据进行重新判断,否则就有可能出现虚假唤醒。
当有多个消费者和生产者时,一次会有多个消费者被唤醒,它们同时执行操作,这也会出现虚假唤醒。应该保证所有的wait操作都在while循环当中。
测试代码
LockTest.java
package juc.lock;
/*
* 虚假唤醒的解决:
* wait要始终保证在while循环当中。
*/
public class LockTest {
public static void main(String[] args) {
Clerk clerk = new Clerk();
Producter producter = new Producter(clerk);
Customer customer = new Customer(clerk);
new Thread(producter,"生产者A").start();
new Thread(customer,"消费者A").start();
new Thread(producter,"生产者B").start();
new Thread(customer,"消费者B").start();
}
}
// 售货员
class Clerk {
private int product = 0;
// 进货
public synchronized void get() {
// 产品已满
while(product >=1) {
System.out.println(Thread.currentThread().getName() + ": " + "已满!");
try {
this.wait();
} catch (InterruptedException e) {
}
}
// 该线程从while中出来的时候,是满足条件的
System.out.println(Thread.currentThread().getName() + ": " + ++product);
this.notifyAll();
}
// 卖货
public synchronized void sale() {
while(product <=0) {
System.out.println(Thread.currentThread().getName() + ": " + "货物不足!");
try {
this.wait();
} catch (InterruptedException e) {
}
}
System.out.println(Thread.currentThread().getName() + ": " + --product);
this.notifyAll();
}
}
// 生产者
class Producter implements Runnable {
private Clerk clerk;
public Producter(Clerk clerk) {
this.clerk = clerk;
}
// 进货
@Override
public void run() {
for(int i = 0; i < 20; ++i) {
try {
Thread.sleep(200);
} catch (InterruptedException e) {
}
clerk.get();
}
}
}
// 消费者
class Customer implements Runnable {
private Clerk clerk;
public Customer(Clerk clerk) {
this.clerk = clerk;
}
// 卖货
@Override
public void run() {
for(int i = 0; i < 20; ++i) {
clerk.sale();
}
}
}
其它
关注下方微信公众号,
回复:
JUC.code
欢迎加入交流群:451826376
更多信息:www.itcourse.top