Producer-Consumer Pattern就是生产者-消费者模式。
生产者和消费者在为不一样的处理线程,生产者必须将数据安全地交给消费者,消费者进行消费时,若是生产者尚未创建数据,则消费者须要等待。
通常来讲,可能存在多个生产者和消费者,不过也有可能生产者和消费者都只有一个,当双方都只有一个时,咱们也称之为Pipe Pattern。安全
该案例中,定义了3个角色:厨师、客人、桌子。dom
厨师(生产者)定义:this
public class MakerThread extends Thread { private final Random random; private final Table table; private static int id = 0; //蛋糕的流水号(全部厨师共通) public MakerThread(String name, Table table, long seed) { super(name); this.table = table; this.random = new Random(seed); } public void run() { try { while (true) { Thread.sleep(random.nextInt(1000)); String cake = "[ Cake No." + nextId() + " by " + getName() + " ]"; table.put(cake); } } catch (InterruptedException e) { } } private static synchronized int nextId() { return id++; } }
客人(消费者)定义:spa
public class EaterThread extends Thread { private final Random random; private final Table table; public EaterThread(String name, Table table, long seed) { super(name); this.table = table; this.random = new Random(seed); } public void run() { try { while (true) { String cake = table.take(); Thread.sleep(random.nextInt(1000)); } } catch (InterruptedException e) { } } }
桌子(队列)定义:线程
public class Table { private final String[] buffer; private int tail; private int head; private int count; public Table(int count) { this.buffer = new String[count]; this.head = 0; this.tail = 0; this.count = 0; } public synchronized void put(String cake) throws InterruptedException { System.out.println(Thread.currentThread().getName() + " puts " + cake); while (count >= buffer.length) { wait(); } buffer[tail] = cake; tail = (tail + 1) % buffer.length; count++; notifyAll(); } public synchronized String take() throws InterruptedException { while (count <= 0) { wait(); } String cake = buffer[head]; head = (head + 1) % buffer.length; count--; notifyAll(); System.out.println(Thread.currentThread().getName() + " takes " + cake); return cake; } }
执行:code
public class Main { public static void main(String[] args) { Table table = new Table(3); new MakerThread("MakerThread-1", table, 31415).start(); new MakerThread("MakerThread-2", table, 92653).start(); new MakerThread("MakerThread-3", table, 58979).start(); new EaterThread("EaterThread-1", table, 32384).start(); new EaterThread("EaterThread-2", table, 62643).start(); new EaterThread("EaterThread-3", table, 38327).start(); } }
Producer-Consumer模式的角色以下:blog
Data表明了实际生产或消费的数据。队列
Producer会建立Data,而后传递给Channel参与者。ip
Consumer从Channel参与者获取Data数据,进行处理。rem
Channel从Producer参与者处接受Data参与者,并保管起来,并应Consumer参与者的要求,将Data参与者传送出去。为确保安全性,Producer参与者与Consumer参与者要对访问共享互斥。