多线程除了我们前面讲的竞争,其实还有协作。就像我们人一样,不但要竞争,也要学会合作,这样才能进步。这篇文章我们就讲讲多线程协作的基本机制wait/notify。同时使用多线程实现生产者/消费者模式。


1.协作的场景

多线程协作的场景有很多,比如:

  • 经典的生产者/消费者模式:生产者消费者通过共享队列实现协作,生产者往队列中放数据,消费者向队列中取数据,当队列满了的时候,生产者就不能再放了,当队列空了的时候,消费者就不能取了。

  • 同时开始:比如百米赛跑,所有的运动员必须等待裁判吹哨子之后才能开始跑。在一些程序,尤其是模拟仿真程序,要求多个线程同时开始。

  • 等待结束:主从协作模式也是一种比较常见的协作模式,主线程将任务分成若干小任务,为每个小任务创建一个线程,主线程必须等待所有子线程都运行结束后才能继续。

  • 集合点:比如班级去春游,在集合点必须等到所有同学都来齐了,才能去下一个旅游点,不能抛下任何一个同学,这是不负责任的。反映在程序中,比如并行迭代计算,每个线程负责一部分计算,然后在集合点等待其他线程完成,一起交付数据。


2.wait/notify方法的介绍

wait/notify方法是类Object中的方法,而Object又是所有类的父类,这样就使得所有的对象都可以调用这两个方法。

主要有两个wait方法

public final void wait() throws InterruptedException
public final void wait(long timeout) throws InterruptedException

不带参数或者参数为0,表示无限期等待。

同时这里说一下wait和sleep的区别:wait会释放对象锁,而且不能主动唤醒,需要其他线程去唤醒它。而sleep不会释放对象锁,且到了设置的时间就会主动唤醒。

前面的文章中有说过,每个对象都有一个锁和一个等待队列,当一个线程尝试去获取对象的锁失败时,就会加入该对象的等待队列。其实对象还有另一个队列,叫条件队列,专门用于线程间的协作。

当一个线程调用wait方法后就会把当前线程加入条件队列中并阻塞,等待一个线程去把它唤醒。唤醒使用notify方法,主要有下面形式:

public final void notify()
public final void notifyAll()

notify方法是在条件队列中随机选择一个线程,将其从条件队列中移除并唤醒。notifyAll方法,顾名思义,就是把条件队列中的线程都唤醒。

wait的具体过程如下:

1.把当前线程放入条件队列,释放对象锁,线程状态变为WAITING或TIMED_WAITING。

2.等待时间到或者其他线程调用notify/notifyAll方法唤醒,从条件队列中移除,但要重新竞争锁。

3.如果获得了对象锁,则线程状态变为RUNNABLE,并从wait方法调用中返回。否则进入等待队列,线程状态变为BLOCKED,只有获得锁之后才从wait方法调用中返回。

说明:线程被唤醒不一定就能立刻获得锁。

ps:wait/notify方法只能在synchronized块中被调用。如果调用wait/notify方法时,当前线程没有获得对象锁,则会抛出异常。


3.生产者/消费者模式

生产者/消费者模式中,协作的共享变量是队列,队列满了,则生产者就wait,队列空了,则消费者就wait。

/**
 * 生产者消费者的共享队列
 */
public class MyBlockingQueue<E> {
    private int length;
    private Queue<E> queue;
    public MyBlockingQueue(int length) {
        this.length = length;
        queue = new ArrayDeque<E>(length);
    }

    public synchronized void put(E e) throws InterruptedException {
        while(queue.size() == length) {
            wait();
        }
        queue.add(e);
        notifyAll();
    }

    public synchronized E get() throws InterruptedException {
        while(queue.isEmpty()) {
            wait();
        }
        E e = queue.poll();
        notifyAll();
        return e;
    }
}

该类是生产者消费者的共享队列,有两个方法,分别是生产者放数据的put方法,以及消费者取数据的get方法。都加了synchronized进行修饰。两个方法中都使用了wait方法,等待的条件不一样,但会加入相同的等待条件队列,所以这里要使用notifyAll方法,因为notify只能唤醒一个线程,如果唤醒的是同类线程那就完蛋了。

只能有一个条件等待队列,这是wait/notify机制的局限性。

/**
 * 生产者线程
 */
public class Producer extends Thread {
    MyBlockingQueue<String> queue;
    public Producer(MyBlockingQueue<String> queue) {
        this.queue = queue;
    }

    @Override
    public void run() {
        int num = 0;
        try {
            while(true) {
                String value = String.valueOf(num);
                queue.put(value);
                System.out.println("producer put " + value);
                num++;
                Thread.sleep((int)(Math.random() * 1000));
            }
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

    }
}
/**
 * 消费者线程
 */
public class Consumer extends Thread {
    MyBlockingQueue<String> queue;
    public Consumer(MyBlockingQueue<String> queue) {
        this.queue = queue;
    }

    @Override
    public void run() {
        try {
            while(true) {
                String value = queue.get();
                System.out.println("Consumer get " + value);
                Thread.sleep((int)(Math.random() * 1000));
            }
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

    }
}

搞一个主程序运行

public class ProducerAndConsumer {
    public static void main(String[] args) {
        MyBlockingQueue<String> queue = new MyBlockingQueue<>(10);
        new Producer(queue).start();
        new Consumer(queue).start();
    }
}

运行后会交替的打印生产者线程和消费者线程的存取信息。

这里我们使用了ArrayDeque,Java提供了专门的阻塞队列实现

  • 接口BlockingQueue和BlockingDeque
  • 基于数组的实现类ArrayBlockingQueue
  • 基于链表的实现类LinkedBlockingQueue和LinkedBlockingDeque
  • 基于堆的实现类PriorityBlockingQueue

在实际开发中,应优先使用这些类。


往期文章
并发基础知识之线程的基本概念
并发基础知识之synchronized关键字