搞定等待通知机制-wait/notify/notifyall的2

2020-11-14  本文已影响0人  Java尖子生

前言

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

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


1、你要知道的基本知识

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

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

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

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

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

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

import java.util.ArrayList;
import java.util.List;
import java.util.UUID;

public class WaitNotifyTest {
    private List<String> 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)]

我这里再整理下。

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

所以说,使用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点:


5、总结

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


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

例:


image.png

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

上一篇 下一篇

猜你喜欢

热点阅读