Lesson_for_java_day20--java的多线程——生产者消费者模式(优化网上生产馒头的案例)

本文主要是介绍Lesson_for_java_day20--java的多线程——生产者消费者模式(优化网上生产馒头的案例),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目的:生产者生产多少产品,消费者就消费多少产品,且生产和消费同时进行。


生产者消费者模式(网上生产馒头的案例):

package sonyi;/* 
* 生产者与消费者模型中,要保证以下几点:  
* 1 同一时间内只能有一个生产者生产     生产方法加锁sychronized  
* 2 同一时间内只能有一个消费者消费     消费方法加锁sychronized  
* 3 生产者生产的同时消费者不能消费     生产方法加锁sychronized  
* 4 消费者消费的同时生产者不能生产     消费方法加锁sychronized  
* 5 共享空间空时消费者不能继续消费     消费前循环判断是否为空,空的话将该线程wait,释放锁允许其他同步方法执行  
* 6 共享空间满时生产者不能继续生产     生产前循环判断是否为满,满的话将该线程wait,释放锁允许其他同步方法执行     
* 
* 生产者消费者模式:对于多个生产者和消费者,为什么要用while判断标记:原因:让被唤醒的线程再一次判断标记。为什么定义notifyAll?因为需要唤醒对方线程。如果只用notify,容易出现只唤醒本方线程,导致程序中的所有线程都在等待。
*/ //主类  
class  ProducerConsumer  
{  public static void main(String[] args)   {  StackBasket s = new StackBasket();  Producer p = new Producer(s);  Consumer c = new Consumer(s);  Thread tp = new Thread(p);  Thread tc = new Thread(c);  tp.start();  tc.start();  }  
}  //产品类  
class Mantou  
{  private int id;  Mantou(int id){  this.id = id;  }  public String toString(){  return "Mantou :" + id;  }  
}  //产品仓库 
class StackBasket  
{  Mantou sm[] = new Mantou[6];  int index = 0;  /**   * show 生产方法.  * show 该方法为同步方法,持有方法锁;  * show 首先循环判断满否,满的话使该线程等待,释放同步方法锁,允许消费;  * show 当不满时首先唤醒正在等待的消费方法,但是也只能让其进入就绪状态,  * show 等生产结束释放同步方法锁后消费才能持有该锁进行消费  * @param m 元素  * @return 没有返回值   */   public synchronized void push(Mantou m){  try{  while(index == sm.length){  System.out.println("!!!!!!!!!生产满了!!!!!!!!!");  this.wait();  }  this.notify();  }catch(InterruptedException e){  e.printStackTrace();  }catch(IllegalMonitorStateException e){  e.printStackTrace();  }  sm[index] = m;  index++;  System.out.println("生产了:" + m + " 仓库有" + index + "个馒头");  }  /**   * show 消费方法  * show 该方法为同步方法,持有方法锁  * show 首先循环判断空否,空的话使该线程等待,释放同步方法锁,允许生产;  * show 当不空时首先唤醒正在等待的生产方法,但是也只能让其进入就绪状态  * show 等消费结束释放同步方法锁后生产才能持有该锁进行生产  * @param b true 表示显示,false 表示隐藏   * @return 没有返回值   */   public synchronized Mantou pop(){  try{  while(index == 0){  System.out.println("!!!!!!!!!消费光了!!!!!!!!!");  this.wait();  }  this.notify();  }catch(InterruptedException e){  e.printStackTrace();  }catch(IllegalMonitorStateException e){  e.printStackTrace();  }  index--;  System.out.println("消费了:---------" + sm[index] + " 仓库还有" + index + "个馒头");  return sm[index];  }  
}  //生产类
class Producer implements Runnable  
{  StackBasket ss = new StackBasket();  Producer(StackBasket ss){  this.ss = ss;  }    public void run(){  for(int i = 1;i <= 20;i++){  Mantou m = new Mantou(i);  ss.push(m);  
//          System.out.println("生产了:" + m + " 共" + ss.index + "个馒头");  
//          在上面一行进行测试是不妥的,对index的访问应该在原子操作里,因为可能在push之后此输出之前又消费了,会产生输出混乱  try{  Thread.sleep((int)(Math.random()*500));  }catch(InterruptedException e){  e.printStackTrace();  }  }  }  
}  //消费类
class Consumer implements Runnable  
{  StackBasket ss = new StackBasket();  Consumer(StackBasket ss){  this.ss = ss;  }  /**   * show 消费进程.  */   public void run(){  //《--问题--》当生产还没结束而消费已经将仓库内产品消费光时,消费线程(循环)就退出,当生产线程将仓库填满时就一直处于等待状态,没办法结束while(ss.index != 0){ ss.pop();  
//          System.out.println("消费了:---------" + m + " 共" + ss.index + "个馒头");  
//  同上在上面一行进行测试也是不妥的,对index的访问应该在原子操作里,因为可能在pop之后此输出之前又生产了,会产生输出混乱  try{  Thread.sleep((int)(Math.random()*1000));  }catch(InterruptedException e){  e.printStackTrace();  }  } System.out.println("消费结束");}  
} 
该案例在运行中有时会出现如下情况:


问题:仓库满后,消费线程却结束了,结果导致生产线程一直处于等待状态。程序没法结束。究其原因是因为消费线程在判断仓库产品为0之后就退出循环,结束消费线程,而不考虑生产线程是否仍在生产。所以当消费线程抢险运行时,就容易结束消费线程(案例中每生产或消费一个产品都会休眠一下,所以生产和消费交替运行,问题不明显,如果去掉休眠,很容易出现问题)。现将生产者消费者模式优化下,在消费线程中增加判断生产线程是否结束,这样就不会出现仓库已满,消费线程却已经结束的结果。


优化后的生产者消费者模式:

package sonyi;public class ProAndConDemo {public static void main(String[] args) {Stack stack = new Stack();new Thread(new Producer1(stack)).start();new Thread(new Consumer1(stack)).start();	}
}class Consumer1 implements Runnable{private Stack stack;public Consumer1(Stack stack) {this.stack = stack;}@Overridepublic void run() {while (Producer1.flag) {//创建标记,如果生产结束,消费才结束stack.pop();//消费产品try {Thread.sleep(1);} catch (InterruptedException e) {e.printStackTrace();}}System.out.println("--------------消费结束--------------");}	
}class Producer1 implements Runnable{public static boolean flag = true;//创建生产标记,开始生产为true,结束生产为falseprivate Stack stack;public Producer1(Stack stack) {this.stack = stack;}@Overridepublic void run() {for(int i = 0; i < 20; i++){//生产20个产品Produce p = new Produce(i+1);//生产一个产品stack.push(p);//产品添加到仓库中try {Thread.sleep(1);} catch (InterruptedException e) {// TODO Auto-generated catch blocke.printStackTrace();}}flag = false;//结束生产标记System.out.println("----------生产结束--------------");}	
}class Stack{Produce[] storeHose = new Produce[5];//创建容量为5的仓库int index = 0;public synchronized void push(Produce p){//往仓库内存放产品(传入一个产品)try {while (index == storeHose.length) {//判断仓库是否已经满了System.out.println("仓库已经满了---------");this.wait();//仓库满时,生产等待}	this.notifyAll();//唤醒其他线程} catch (InterruptedException e) {e.printStackTrace();}storeHose[index++] = p;//仓库添加一个产品System.out.println("生产了" + p.id + "号产品,现在仓库有:" + index + "产品");	}public synchronized void pop(){	//往仓库中消费产品try {//双重判断,只有当仓库没有产品和成产还在继续时,消费线程才进入等待。//也就是说,当生产还在继续时,仓库没有产品,消费线程才进入等待,当生产结束时,消费线程不会进入等待,而是将产品消费光while(index == 0 && Producer1.flag){System.out.println("产品已经消费完了---------");this.wait();}this.notifyAll();//消费线程等待时,唤醒其他线程} catch (InterruptedException e) {e.printStackTrace();}	Produce p = storeHose[--index];//消费仓库里的一个产品System.out.println("消费了" + p.id + "号产品,现在仓库还有" + index + "产品");	}
}//创建产品类,每个产品都有自己的id
class Produce{int id;public Produce(int id) {this.id = id;}
}

欢迎java学习者共同讨论。

这篇关于Lesson_for_java_day20--java的多线程——生产者消费者模式(优化网上生产馒头的案例)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/365291

相关文章

Java实现检查多个时间段是否有重合

《Java实现检查多个时间段是否有重合》这篇文章主要为大家详细介绍了如何使用Java实现检查多个时间段是否有重合,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录流程概述步骤详解China编程步骤1:定义时间段类步骤2:添加时间段步骤3:检查时间段是否有重合步骤4:输出结果示例代码结语作

Java中String字符串使用避坑指南

《Java中String字符串使用避坑指南》Java中的String字符串是我们日常编程中用得最多的类之一,看似简单的String使用,却隐藏着不少“坑”,如果不注意,可能会导致性能问题、意外的错误容... 目录8个避坑点如下:1. 字符串的不可变性:每次修改都创建新对象2. 使用 == 比较字符串,陷阱满

Java判断多个时间段是否重合的方法小结

《Java判断多个时间段是否重合的方法小结》这篇文章主要为大家详细介绍了Java中判断多个时间段是否重合的方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录判断多个时间段是否有间隔判断时间段集合是否与某时间段重合判断多个时间段是否有间隔实体类内容public class D

IDEA编译报错“java: 常量字符串过长”的原因及解决方法

《IDEA编译报错“java:常量字符串过长”的原因及解决方法》今天在开发过程中,由于尝试将一个文件的Base64字符串设置为常量,结果导致IDEA编译的时候出现了如下报错java:常量字符串过长,... 目录一、问题描述二、问题原因2.1 理论角度2.2 源码角度三、解决方案解决方案①:StringBui

Java覆盖第三方jar包中的某一个类的实现方法

《Java覆盖第三方jar包中的某一个类的实现方法》在我们日常的开发中,经常需要使用第三方的jar包,有时候我们会发现第三方的jar包中的某一个类有问题,或者我们需要定制化修改其中的逻辑,那么应该如何... 目录一、需求描述二、示例描述三、操作步骤四、验证结果五、实现原理一、需求描述需求描述如下:需要在

Java中ArrayList和LinkedList有什么区别举例详解

《Java中ArrayList和LinkedList有什么区别举例详解》:本文主要介绍Java中ArrayList和LinkedList区别的相关资料,包括数据结构特性、核心操作性能、内存与GC影... 目录一、底层数据结构二、核心操作性能对比三、内存与 GC 影响四、扩容机制五、线程安全与并发方案六、工程

JavaScript中的reduce方法执行过程、使用场景及进阶用法

《JavaScript中的reduce方法执行过程、使用场景及进阶用法》:本文主要介绍JavaScript中的reduce方法执行过程、使用场景及进阶用法的相关资料,reduce是JavaScri... 目录1. 什么是reduce2. reduce语法2.1 语法2.2 参数说明3. reduce执行过程

如何使用Java实现请求deepseek

《如何使用Java实现请求deepseek》这篇文章主要为大家详细介绍了如何使用Java实现请求deepseek功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1.deepseek的api创建2.Java实现请求deepseek2.1 pom文件2.2 json转化文件2.2

Java调用DeepSeek API的最佳实践及详细代码示例

《Java调用DeepSeekAPI的最佳实践及详细代码示例》:本文主要介绍如何使用Java调用DeepSeekAPI,包括获取API密钥、添加HTTP客户端依赖、创建HTTP请求、处理响应、... 目录1. 获取API密钥2. 添加HTTP客户端依赖3. 创建HTTP请求4. 处理响应5. 错误处理6.

Spring AI集成DeepSeek的详细步骤

《SpringAI集成DeepSeek的详细步骤》DeepSeek作为一款卓越的国产AI模型,越来越多的公司考虑在自己的应用中集成,对于Java应用来说,我们可以借助SpringAI集成DeepSe... 目录DeepSeek 介绍Spring AI 是什么?1、环境准备2、构建项目2.1、pom依赖2.2