搞定等待通知机制-wait/notify/notifyall的2个经典面试题(实例详解)

前言

关于wait/notify/notifyall有2个经典的面试:

  • notify和notifyall有什么区别?
  • 为什么wait方法要写在while循环里面而不是if呢?

带着这2个问题,我们来学习下synchronized提供的等待通知机制。



1、你要知道的基本知识

  • wait/notify/notifyall都是存在与Object类里面的方法。

ps:为什么要存在Object里面而不是Thread里面呢?赶紧看下这篇吧(大彻大悟synchronized原理,锁的升级),从底层了解其原因。

  • wait方法:wait方法只是无参和带超时时间2种方法,调用wait方法的的线程会进入waiting或timed_waiting状态;
  • notify方法:调用notify方法的线程,会唤醒一个处于waiting状态的线程;
  • notifyall方法:调用notify方法的线程,会唤醒所有处于waiting状态的线程;

有一个前提,就是wait/notify/notifyall方法必须在获取到synchronized资源锁的情况下,才能调用,也就是wait/notify/notifyall必须在synchronized代码块里面。

使用wait/notify/notifyal有个经典的范式,这便是上面的第二个问题。

  while(条件不满足) {
    wait();
  }



2、典型的生产者消费者模式的样例

现在我们写个典型的生产者消费者模式的样例:

  • 有2个生产者,当list不满的情况下,往list里面添加数据,当list满的情况下调用wait方法。每添加一条数据,就调用notifyAll()方法。
  • 2个消费者,当list不为空的情况下,消费list里面的第一个元素,并且调用notifyAll()方法。
  • 为了效果,我让每个生产者只生产了3条数据,消费者一直在消费,最终都进入waiting状态。
import java.util.ArrayList;
import java.util.List;
import java.util.UUID;

public class WaitNotifyTest {
    private List list = new ArrayList<>();

    public static void main(String[] args) {
        WaitNotifyTest waitNotifyTest = new WaitNotifyTest();
        Thread producer1 = new Thread(() -> {
            for (int i = 0; i < 3; i++) {
                waitNotifyTest.produce();
            }
        });
        producer1.setName("生产者1号");
        Thread producer2 = new Thread(() -> {
            for (int i = 0; i < 3; i++) {
                waitNotifyTest.produce();
            }
        });
        producer2.setName("生产者2号");

        Thread consumer1 = new Thread(() -> {
            while (true) waitNotifyTest.consume();
        });
        consumer1.setName("消费者1号");
        Thread consumer2 = new Thread(() -> {
            while (true) waitNotifyTest.consume();
        });
        consumer2.setName("消费者2号");

        producer1.start();
        producer2.start();
        consumer1.start();
        consumer2.start();
    }

    private void produce() {
        synchronized (this) {
            while (listIsFull()) {
                try {
                    System.out.println(Thread.currentThread().getName() + "进入等待池");
                    wait();
                    System.out.println(Thread.currentThread().getName() + "被唤醒了");
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            String value = UUID.randomUUID().toString();
            System.out.println(Thread.currentThread().getName() + "生产了一条消息:" + value);
            list.add(value);
            notifyAll();
        }
    }

    private void consume() {
        synchronized (this) {
            while (listIsEmpty()) {
                try {
                    System.out.println(Thread.currentThread().getName() + "进入等待池");
                    wait();
                    System.out.println(Thread.currentThread().getName() + "被唤醒了");
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            System.out.println(Thread.currentThread().getName() + "消费了一条消息:" + list.get(0));
            list.remove(0);
            notifyAll();
        }
    }

    private boolean listIsFull() {
        return list.size() == 1;
    }

    private boolean listIsEmpty() {
        return list.size() == 0;
    }
}

执行结果:

生产者1号生产了一条消息:85158920-3784-4c5f-9c2a-2961aea75086
消费者2号消费了一条消息:85158920-3784-4c5f-9c2a-2961aea75086
消费者2号进入等待池
生产者2号生产了一条消息:3caf8352-28d3-44f6-8a7f-5ecf7356e903
生产者2号进入等待池
消费者1号消费了一条消息:3caf8352-28d3-44f6-8a7f-5ecf7356e903
消费者1号进入等待池
生产者2号被唤醒了
生产者2号生产了一条消息:b8764c83-8b42-4527-a953-4424e1103111
生产者2号进入等待池
消费者2号被唤醒了
消费者2号消费了一条消息:b8764c83-8b42-4527-a953-4424e1103111
消费者2号进入等待池
生产者1号生产了一条消息:32228e02-d7f1-4866-83c7-d3fcfb6a51b4
生产者1号进入等待池
消费者2号被唤醒了
消费者2号消费了一条消息:32228e02-d7f1-4866-83c7-d3fcfb6a51b4
消费者2号进入等待池
生产者2号被唤醒了
生产者2号生产了一条消息:3af2d730-7bf3-4a26-8a96-f6f3f30fc39a
消费者1号被唤醒了
消费者1号消费了一条消息:3af2d730-7bf3-4a26-8a96-f6f3f30fc39a
消费者1号进入等待池
消费者2号被唤醒了
消费者2号进入等待池
生产者1号被唤醒了
生产者1号生产了一条消息:0867b9ae-de1a-486b-b0b0-e56c6b10a49b
消费者2号被唤醒了
消费者2号消费了一条消息:0867b9ae-de1a-486b-b0b0-e56c6b10a49b
消费者2号进入等待池
消费者1号被唤醒了
消费者1号进入等待池



3、notify和notifyAll有什么区别

最简单的回答就是:notify只唤醒一个waiting状态的线程,notifyAll唤醒所有waiting状态的线程

但是要彻底弄清楚它们的区别,还是要从synchronized的底层说起。看过这篇文章的显然已经知道答案了。
[图片上传失败...(image-2d12eb-1605322860624)]

我这里再整理下。

  • synchronized维护的对象锁有2个队列,一个_EntryList,一个_WaitSet。
  • 加锁时,线程获取到锁进入临界区(_owner),若线程获取不到锁,便加入_EntryList,进入blocked阻塞状态。
  • 线程获取到锁后,调用wait方法,被加入_WaitSet队列,进入waiting状态,然后等待唤醒。当线程被唤醒的时候,被唤醒的线程需要再次获取对象锁
  • 唤醒线程,我们可以调用notify和notifyAll方法,notify只是随机的唤醒一个_WaitSet中的线程。notifyAll会唤醒所有处于_WaitSet中的线程。
  • 不管唤醒一个线程,还是唤醒多个线程,最终获得对象锁的,只有一个线程。如果_EntryList同时存在竞争锁资源的线程,那么被唤醒的线程还需要和_EntryList中的线程一起竞争锁资源。但是JVM保证最终只会让一个线程获取到锁

那如果只唤醒一个线程会有什么问题呢?
拿上面的生产者消费者举个例子:

  • 当list为空时,消费者consumer1、consumer2都会处于waiting状态
  • 生产者producer1和生产者producer2竞争锁,生产者producer1先拿到锁,生产一条数据,调用notify(假设唤醒的是消费者consumer1),然后producer1进入阻塞blocked状态,并释放锁;
  • 消费者consumer1被唤醒后,有三个线程同时竞争锁(producer1、producer2、consumer1),假设producer2获得锁,producer2发现list满了,然后进入waiting状态,并释放锁;
  • 锁被释放后,有两个个线程同时竞争锁(producer1、consumer1),假设consumer1获取到锁,consumer1消费消息,然后调用notify(此时consumer2和生产者producer2处于waiting);
    问题就出在这里:
    假设唤醒的是生产者producer2,没有问题;
    假设唤醒的是消费者consumer2,那consumer2会发现list任然为空,继续进入waiting状态。但是呢,恰好之前进入阻塞状态的producer1已经下线(在这个例子中就是生产了3条数据)。这样就出现了死锁了(producer2和consumer2都处于waiting状态,消费者consumer1在获取到锁后也会进入wait状态)

所以说,使用notify某些希望被唤醒的线程,永远得不到唤醒,获取不到锁资源,导致死锁。

综上所述:我们尽量使用notifyAll而不是notify。除非你经过深思熟虑,且明确知道唤醒的就你希望的线程(比如上例中只有一个生产者一个消费者)



4、为什么wait方法要写在while循环里面而不是if呢

再来看下第二个问题:为什么wait方法要写在while循环里面而不是if呢?

private void produce() {
    synchronized (this) {
        while (listIsFull()) {
            try {
                System.out.println(Thread.currentThread().getName() + "进入等待池");
                wait();
                System.out.println(Thread.currentThread().getName() + "被唤醒了");
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
        String value = UUID.randomUUID().toString();
        System.out.println(Thread.currentThread().getName() + "生产了一条消息:" + value);
        list.add(value);
        notifyAll();
    }
}

明确这2点:

  • 线程被唤醒之后,代码是紧接着执行wait后面的代码(从上面的执行结果可以看出);
  • 进入waiting状态的线程被唤醒的条件是“条件满足”,对应到下面的例子就是队列不满。
    被唤醒的线程需要和其他线程竞争锁资源(最终只有一个线程获取到),那么当被唤醒的线程获得锁资源的时候,之前的条件可能又不满足了。
    如果while改成if,那么被唤醒的线程继续执行(默认条件任然满足),这明显会导致并发问题,比如超额生产、消费。



5、总结

以后遇到同样的问题知道怎么回答了吧!


ps:一天一个IDEA小技巧
快捷键[Alt+7]可以打开当前类的架构图(Structure),可以快速查看类、方法、字段等。这样可以提升工作效率哦,不要用鼠标滚动查找啦!!!

例:


image.png

多线程连载:
Java内存模型-volatile的应用(实例讲解)
synchronized的三种应用方式(实例讲解)
可重入锁-synchronized是可重入锁吗?
大彻大悟synchronized原理,锁的升级
一文弄懂Java的线程池
公平锁和非公平锁-ReentrantLock是如何实现公平、非公平的
一图全面了解Java线程的生命周期
守护线程和用户线程的真正区别(实例讲解)

你可能感兴趣的:(搞定等待通知机制-wait/notify/notifyall的2个经典面试题(实例详解))