JavaSE-20.3生产者消费者

Posted yub4by

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了JavaSE-20.3生产者消费者相关的知识,希望对你有一定的参考价值。

 

  1 package day11.lesson3;
  2 
  3 /*
  4 3 生产者消费者
  5 
  6 3.1 生产者消费者模式
  7 
  8     生产者消费者模式是一个十分经典的多线程协作的模式,弄懂生产者消费者问题能够让我们对多线程编程的理解更加深刻。
  9     所谓生产者消费者问题,实际上主要是包含了两类线程:
 10         一类是生产者线程用于生产数据
 11         一类是消费者线程用于消费数据
 12     为了解耦生产者和消费者的关系,通常会采用共享的数据区域,就像是一个仓库(见图)
 13         生产者生产数据之后直接放置在共享数据区中,并不需要关心消费者的行为
 14         消费者只需要从共享数据区中去获取数据,并不需要关心生产者的行为
 15 
 16     Object类的等待和唤醒方法
 17         void wait() 导致当前线程等待,直到另一个线程调用该对象的 notify()方法或 notifyAll()方法
 18         void notify() 唤醒正在等待对象监视器的单个线程
 19         void notifyAll() 唤醒正在等待对象监视器的所有线程
 20 
 21 3.2 生产者消费者案例
 22 
 23     生产者消费者案例中包含的类:
 24         奶箱类(Box):定义一个成员变量,表示第x瓶奶,提供存储牛奶和获取牛奶的操作
 25         生产者类(Producer):实现Runnable接口,重写run()方法,调用存储牛奶的操作
 26         消费者类(Customer):实现Runnable接口,重写run()方法,调用获取牛奶的操作
 27         测试类(BoxDemo):里面有main方法,main方法中的代码步骤如下
 28             ①创建奶箱对象,这是共享数据区域
 29             ②创建消费者创建生产者对象,把奶箱对象作为构造方法参数传递,因为在这个类中要调用存储牛奶的操作
 30             ③对象,把奶箱对象作为构造方法参数传递,因为在这个类中要调用获取牛奶的操作
 31             ④创建2个线程对象,分别把生产者对象和消费者对象作为构造方法参数传递
 32             ⑤启动线程
 33  */
 34 public class BoxDemo {
 35     public static void main(String[] args) {
 36         Box b = new Box();
 37         Producer p = new Producer(b);
 38         Customer c = new Customer(b);
 39 
 40         Thread t1 = new Thread(p);
 41         Thread t2 = new Thread(c);
 42 
 43         t1.start();
 44         t2.start();
 45     }
 46 }
 47 
 48 
 49 class Box{
 50     private int milk; //第几瓶奶
 51     private boolean state = false; //奶箱状态
 52 
 53     public synchronized void put(int milk){
 54         //奶箱中若有牛奶,则等待消费
 55         if (state){
 56             try {
 57                 wait(); //
 58             } catch (InterruptedException e) {
 59                 e.printStackTrace();
 60             }
 61         }
 62 
 63         //奶箱中若无牛奶,则生产牛奶
 64         this.milk = milk;
 65         System.out.println("送奶工将第" + this.milk + "瓶奶放入奶箱");
 66 
 67         //生产牛奶完毕后,修改奶箱状态
 68         state = true;
 69 
 70         //唤醒其他等待的线程
 71         notifyAll();
 72     }
 73 
 74     public synchronized void get(){
 75         //奶箱中若无牛奶,则等待生产
 76         if (!state){
 77             try {
 78                 wait(); //
 79             } catch (InterruptedException e) {
 80                 e.printStackTrace();
 81             }
 82         }
 83 
 84         //奶箱中若有牛奶,则等待消费
 85         System.out.println("用户拿到第" + this.milk + "瓶奶");
 86 
 87         //消费牛奶完毕后,修改奶箱状态
 88         state = false;
 89 
 90         //唤醒其他等待的线程
 91         notifyAll();
 92     }
 93 }
 94 
 95 
 96 class Producer implements Runnable{
 97     private Box b;
 98 
 99     public Producer(Box b) {
100         this.b = b;
101     }
102 
103     @Override
104     public void run() {
105         for (int i=1; i<=5; i++){
106             b.put(i);
107         }
108     }
109 }
110 
111 
112 class Customer implements Runnable{
113     private Box b;
114 
115     public Customer(Box b) {
116         this.b = b;
117     }
118 
119     @Override
120     public void run() {
121         while (true){
122             b.get();
123         }
124     }
125 }

 

以上是关于JavaSE-20.3生产者消费者的主要内容,如果未能解决你的问题,请参考以下文章

RocketMQ - 如何用死信队列解决消费者异常

请问如何用C语言实现“生产者与消费者问题”?(最好附上完整的C语言源代码)

生产者消费者模型-Java代码实现

生产者和消费者模式-代码

golang生产者消费者模型示例代码

golang生产者消费者模型示例代码