生产者消费者问题

生产者消费者问题

 

(01) 生产者仅仅在仓储未满时候生产,仓满则停止生产。

(02) 消费者仅仅在仓储有产品时候才能消费,仓空则等待。

(03) 当消费者发现仓储没产品可消费时候会通知生产者生产。

(04) 生产者在生产出可消费产品时候,应该通知等待的消费者去消费。

代码示例:

  1 package thread.Test;
  2 
  3 //Demo1.java
  4 //仓库
  5 class Depot {
  6  private int capacity;    // 仓库的容量
  7  private int size;        // 仓库的实际数量
  8 
  9  public Depot(int capacity) {
 10      this.capacity = capacity;
 11      this.size = 0;
 12  }
 13 
 14  public synchronized void produce(int val) {
 15      try {
 16           // left 表示“想要生产的数量”(有可能生产量太多,需多此生产)
 17          int left = val;
 18          while (left > 0) {
 19              // 库存已满时,等待“消费者”消费产品。
 20              while (size >= capacity)
 21                  wait();
 22              // 获取“实际生产的数量”(即库存中新增的数量)
 23              // 如果“库存”+“想要生产的数量”>“总的容量”,则“实际增量”=“总的容量”-“当前容量”。(此时填满仓库)
 24              // 否则“实际增量”=“想要生产的数量”
 25              int inc = (size+left)>capacity ? (capacity-size) : left;
 26              size += inc;
 27              left -= inc;
 28              System.out.printf("%s produce(%3d) --> left=%3d, inc=%3d, size=%3d\n", 
 29                      Thread.currentThread().getName(), val, left, inc, size);
 30              // 通知“消费者”可以消费了。
 31              notifyAll();
 32          }
 33      } catch (InterruptedException e) {
 34      }
 35  } 
 36 
 37  public synchronized void consume(int val) {
 38      try {
 39          // left 表示“客户要消费数量”(有可能消费量太大,库存不够,需多此消费)
 40          int left = val;
 41          while (left > 0) {
 42              // 库存为0时,等待“生产者”生产产品。
 43              while (size <= 0)
 44                  wait();
 45              // 获取“实际消费的数量”(即库存中实际减少的数量)
 46              // 如果“库存”<“客户要消费的数量”,则“实际消费量”=“库存”;
 47              // 否则,“实际消费量”=“客户要消费的数量”。
 48              int dec = (size<left) ? size : left;
 49              size -= dec;
 50              left -= dec;
 51              System.out.printf("%s consume(%3d) <-- left=%3d, dec=%3d, size=%3d\n", 
 52                      Thread.currentThread().getName(), val, left, dec, size);
 53              notifyAll();
 54          }
 55      } catch (InterruptedException e) {
 56      }
 57  }
 58 
 59  public String toString() {
 60      return "capacity:"+capacity+", actual size:"+size;
 61  }
 62 } 
 63 
 64 //生产者
 65 class Producer {
 66  private Depot depot;
 67  
 68  public Producer(Depot depot) {
 69      this.depot = depot;
 70  }
 71 
 72  // 消费产品:新建一个线程向仓库中生产产品。
 73  public void produce(final int val) {
 74      new Thread() {
 75          public void run() {
 76              depot.produce(val);
 77          }
 78      }.start();
 79  }
 80 }
 81 
 82 //消费者
 83 class Customer {
 84  private Depot depot;
 85  
 86  public Customer(Depot depot) {
 87      this.depot = depot;
 88  }
 89 
 90  // 消费产品:新建一个线程从仓库中消费产品。
 91  public void consume(final int val) {
 92      new Thread() {
 93          public void run() {
 94              depot.consume(val);
 95          }
 96      }.start();
 97  }
 98 }
 99 
100 public class ConsumerProductTest {  
101  public static void main(String[] args) {  
102      Depot mDepot = new Depot(100);
103      Producer mPro = new Producer(mDepot);
104      Customer mCus = new Customer(mDepot);
105 
106      mPro.produce(60);
107      mPro.produce(120);
108      mCus.consume(90);
109      mCus.consume(150);
110      mPro.produce(110);
111  }
112 }

控制台输出:

Thread-1 produce(120) --> left= 20, inc=100, size=100
Thread-3 consume(150) <-- left= 50, dec=100, size=  0
Thread-4 produce(110) --> left= 10, inc=100, size=100
Thread-2 consume( 90) <-- left=  0, dec= 90, size= 10
Thread-3 consume(150) <-- left= 40, dec= 10, size=  0
Thread-1 produce(120) --> left=  0, inc= 20, size= 20
Thread-3 consume(150) <-- left= 20, dec= 20, size=  0
Thread-0 produce( 60) --> left=  0, inc= 60, size= 60
Thread-4 produce(110) --> left=  0, inc= 10, size= 70
Thread-3 consume(150) <-- left=  0, dec= 20, size= 50

 

posted @ 2017-03-13 23:29  乱码出黑客  阅读(221)  评论(0编辑  收藏  举报