描述

生产消费问题在java多线程的学习中是经常遇到的问题 ,多个线程共享通一个资源的时候会出现各种多线程中经常出现的各种问题。

实例说明

三个类:售货员Clerk,工厂Factory,消费者Consumer

Factory和Consumer共享Clerk对象

1.普通情况

  1. Clerk类:
class Clerk{
//商品数量默认是0,volatile关键字保证内存可见性
private volatile int product=0; //进货,synchronized关键字保证原子性,互斥性
public synchronized void get(){
if(product>10){
System.out.println("货满了");
}else {
++product;
System.out.println(Thread.currentThread().getName()+"进货"+product);
}
} //售货
public synchronized void sale(){
if(product<=0){
System.out.println("没货了");
}else{
System.out.println(Thread.currentThread().getName()+"卖货"+product);
--product;
}
}
}
  1. Factory类:
class Factory implements Runnable{

    private Clerk clerk;

    Factory(Clerk clerk){
this.clerk = clerk;
} @Override
public void run() {
for(int i=0;i<20;i++){
clerk.get();//进货
}
} }
  1. Consumer类:
class Consumer implements Runnable{

    private Clerk clerk;

    Consumer(Clerk clerk){
this.clerk = clerk;
} @Override
public void run() {
for(int i=0;i<20;i++){
clerk.sale();//卖货
} }
}

测试:

public static void main(String[] args) {

        Clerk clerk = new Clerk();
Factory factory = new Factory(clerk);
Consumer consumer = new Consumer(clerk); Thread tf = new Thread(factory);
Thread tc = new Thread(consumer); tf.start();
tc.start(); }

输出结果:

Thread-0进货1
Thread-0进货2
Thread-0进货3
Thread-0进货4
Thread-0进货5
Thread-0进货6
Thread-0进货7
Thread-0进货8
Thread-0进货9
Thread-0进货10
Thread-0进货11
货满了
货满了
货满了
货满了
货满了
货满了
货满了
货满了
货满了
Thread-1卖货11
Thread-1卖货10
Thread-1卖货9
Thread-1卖货8
Thread-1卖货7
Thread-1卖货6
Thread-1卖货5
Thread-1卖货4
Thread-1卖货3
Thread-1卖货2
Thread-1卖货1
没货了
没货了
没货了
没货了
没货了
没货了
没货了
没货了
没货了

问题出现了,每次进货只有在进货满了的情况下,才会买货,当进货的次数执行完了之后才会执行卖货的方法,而且卖货没货的时候一直输出没货不会等待商家进货。

1. 重复调用占用资源问题

原因分析 :

上述的情况是当没货的时候还会继续调用该方法,从而占用资源,二货满的情况下也会重复调用进货方法,占用资源,这样是不合理的。

解决方式:

当货满了,应该停止进货,释放锁让消费者消费,当没货了应该停止消费释放锁,让进货,这是我们想要的逻辑。

使用wait()notifyAll()这两个方法来实现。

修改Clerk的get和sale方法如下:

class Clerk{
//商品数量默认是0
private volatile int product=0; //进货
public synchronized void get(){
if(product>10){
System.out.println("货满了");
try {
this.wait();//等待并释放clerk的对象锁,进入线程队列等待被唤醒
} catch (InterruptedException e) {
e.printStackTrace();
}
}else {
++product;
System.out.println(Thread.currentThread().getName()+"进货"+product);
notifyAll();//唤醒等待的线程
}
} //售货
public synchronized void sale(){
if(product<=0){
System.out.println("没货了");
try {
this.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}else{
System.out.println(Thread.currentThread().getName()+"卖货"+product);
--product;
notifyAll();
}
}
}

输出测试:

Thread-0进货1
Thread-1卖货1
没货了
Thread-0进货1
Thread-0进货2
Thread-0进货3
Thread-0进货4
Thread-0进货5
Thread-0进货6
Thread-0进货7
Thread-0进货8
Thread-0进货9
Thread-0进货10
Thread-0进货11
货满了
Thread-1卖货11
Thread-1卖货10
Thread-1卖货9
Thread-1卖货8
Thread-1卖货7
Thread-1卖货6
Thread-1卖货5
Thread-1卖货4
Thread-1卖货3
Thread-1卖货2
Thread-1卖货1
没货了
Thread-0进货1
Thread-0进货2
Thread-0进货3
Thread-0进货4
Thread-0进货5
Thread-0进货6
Thread-0进货7
Thread-1卖货7
Thread-1卖货6
Thread-1卖货5
Thread-1卖货4
Thread-1卖货3
Thread-1卖货2

这样看起来和谐多了,但是还存在一个小问题,那就是当商品数量变少的时候,而且Factory或者Consumer的run方法内Thread.sleep()方法进行延时,在真是的项目中,这中延时是真实存在的。会产生一方提前结束了,而另外一方没有被唤醒的的情况,从而导致线程一直在等待无法结束的情况产生。

2. 线程阻塞无法唤醒

当product比较小假如是1的时候,有可能生产者先循环结束,

消费者还没结束,一直在waite无法得到唤醒就一直等待

程序就会停在那里

解决方式:去掉else,保证每次都会唤醒另外一个线程

//店员
class Clerk{
private int product;
private volatile boolean proFlg=true;//生产者是否完结的标志位
public boolean isProFlg() {
return proFlg;
} public void setProFlg(boolean proFlg) {
this.proFlg = proFlg;
}
public synchronized void addProduct(){
while(product>=10){
try {
wait();//大于10各产品,停止生产
} catch (InterruptedException e) {
e.printStackTrace();
}
}//else{
product++;
System.out.println(Thread.currentThread().getName()+"生产:"+product);
notifyAll();
//}
} public synchronized void saleProduct(){
while(product <= 0 && !this.isProFlg()){
try {
wait();
} catch (InterruptedException e) {
e.printStackTrace();
}//大于10各产品,停止消费
}//else {
System.out.println(Thread.currentThread().getName()+"消费:"+product);
product--;
notifyAll();
//}
} }

3.虚假唤醒问题

Clerk clerk = new Clerk();
Factory factory = new Factory(clerk);
Consumer consumer = new Consumer(clerk); Thread tf = new Thread(factory);
Thread tc = new Thread(consumer);
Thread tc2 = new Thread(consumer);
tf.start();
tc.start();
tc2.start();

输出结果:

没货了
没货了
Thread-0进货1
Thread-2卖货1
没货了
Thread-1卖货0
没货了
Thread-2卖货-1
没货了
Thread-1卖货-2
没货了
Thread-2卖货-3
没货了
Thread-1卖货-4
没货了
Thread-2卖货-5
没货了
Thread-1卖货-6

当只有一个Factory有两个Consumer的时候就会出现虚假唤醒问题。导致商品都成了负数了。

原因分析:

当创建对个生产消费者线程的时候,会产生虚假唤醒,导致product

为负数,是因为当消费者线程A发现没货的时候,wait之后释放锁,另外一个

消费者线程B获得锁开始执行,结果也没货,开始wait,当生产者生产之后

notifyAll,A,B线程开始继续向下执行,结果进行了两次–操作,导致

product成为了负数

解决方式:

JDK文档object的wait方法已经考虑到这种情况,防止虚假唤醒,应该放在循环中,多次进行检查,直到满足条件才进行下一步

class Clerk{
//商品数量默认是0
private volatile int product=0; //进货
public synchronized void get(){
while(product>10){
System.out.println("货满了");
try {
this.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
++product;
System.out.println(Thread.currentThread().getName()+"进货"+product);
notifyAll(); } //售货
public synchronized void sale(){
while(product<=0){
System.out.println("没货了");
try {
this.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
System.out.println(Thread.currentThread().getName()+"卖货"+product);
--product;
notifyAll(); }
}

4. 守护线程解决线程阻塞

上面解决了虚假唤醒问题,但是当多个消费者和一个生产者的时候,生产者有可能先结束循环,但是消费者还没结束,结果到了其他消费者的时候发现product是小于0的于是就wait,程序一直等待得不到结束,就会一直在wait()

解决方式:

在共享资源clerk类中定义生产者线程标志位,在main线程中创建一个线程设置为守护线程

并启动,在该守护线程中创建匿名内部类Runnable并在run方法中判断生产者线程isAlive()

如果生产者线程结束,就把标志位置为false,该标识位和消费者线程的while判断条件中串联

当生产者线程为false的之后短路,使得消费者线程啥都不做,直到线程结束。

  1. Clerk中设置Factory线程的标志位
private boolean facctoryFlg = true;//工厂线程结束的标志位,为false表示线程执行完毕
public boolean isFacctoryFlg() {
return facctoryFlg;
} public void setFacctoryFlg(boolean facctoryFlg) {
this.facctoryFlg = facctoryFlg;
}
  1. 主方法中创建守护线程
//创建守护线程
Thread daemon = new Thread(new Runnable() {
@Override
public void run() {
while(true){
if(!tf.isAlive()){
clerk.setFacctoryFlg(false);
System.out.println("factory--------------"+tf.isAlive());
break;
}
}
}
});
daemon.setDaemon(true);//设置为守护线程(后台线程)
daemon.start();
  1. 修改Clerk的sale方法:
//售货
public synchronized void sale(){ while(product<=0){
//当Factory线程结束的时候,直接结束sale方法
if(!isFacctoryFlg()){
return;
}
System.out.println("没货了");
try {
this.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
System.out.println(Thread.currentThread().getName()+"卖货"+product);
--product;
notifyAll();
}

通过守护线程daemon的监视,可以避免线程阻塞的情况,就算有多个消费者或者Factory只要在守护线程中添加判断逻辑,就可以避免阻塞的出现。

7.JUC线程高级-生产消费问题&虚假唤醒的更多相关文章

  1. (二)juc线程高级特性——CountDownLatch / Callable / Lock

    5. CountDownLatch 闭锁 Java 5.0 在 java.util.concurrent 包中提供了多种并发容器类来改进同步容器的性能. CountDownLatch 一个同步辅助类, ...

  2. (四)juc线程高级特性——线程池 / 线程调度 / ForkJoinPool

    13. 线程池 第四种获取线程的方法:线程池,一个 ExecutorService,它使用可能的几个池线程之一执行每个提交的任务,通常使用 Executors 工厂方法配置. 线程池可以解决两个不同问 ...

  3. (一)juc线程高级特性——volatile / CAS算法 / ConcurrentHashMap

    1. volatile 关键字与内存可见性 原文地址: https://www.cnblogs.com/zjfjava/category/979088.html 内存可见性(Memory Visibi ...

  4. Java多线程中的虚假唤醒和如何避免

    先来看一个例子 一个卖面的面馆,有一个做面的厨师和一个吃面的食客,需要保证,厨师做一碗面,食客吃一碗面,不能一次性多做几碗面,更不能没有面的时候吃面:按照上述操作,进行十轮做面吃面的操作. 用代码说话 ...

  5. (三)juc高级特性——虚假唤醒 / Condition / 按序交替 / ReadWriteLock / 线程八锁

    8. 生产者消费者案例-虚假唤醒 参考下面生产者消费者案例: /* * 生产者和消费者案例 */ public class TestProductorAndConsumer { public stat ...

  6. JUC虚假唤醒(六)

    为什么条件锁会产生虚假唤醒现象(spurious wakeup)? ​ 在不同的语言,甚至不同的操作系统上,条件锁都会产生虚假唤醒现象.所有语言的条件锁库都推荐用户把wait()放进循环里: whil ...

  7. kafka生产消费原理笔记

    一.什么是kafka Kafka是最初由Linkedin公司开发,是一个分布式.支持分区的(partition).多副本的(replica),基于zookeeper协调的分布式消息系统,它的最大的特性 ...

  8. 线程高级篇-Lock锁实现生产者-消费者模型

    Lock锁介绍: 在java中可以使用 synchronized 来实现多线程下对象的同步访问,为了获得更加灵活使用场景.高效的性能,java还提供了Lock接口及其实现类ReentrantLock和 ...

  9. 第44天学习打卡(JUC 线程和进程 并发和并行 Lock锁 生产者和消费者问题 如何判断锁(8锁问题) 集合类不安全)

    什么是JUC 1.java.util工具包 包 分类 业务:普通的线程代码 Thread Runnable 没有返回值.效率相比Callable相对较低 2.线程和进程 进程:一个程序.QQ.exe, ...

随机推荐

  1. Prthon多线程和模块

    Prthon多线程和模块 案例1:简化除法判断 案例2:分析apache访问日志 案例3:扫描存活主机 案例4:利用多线程实现ssh并发访问 1 案例1:简化除法判断 1.1 问题 编写mydiv.p ...

  2. Linux/Unix下pid文件的作用

    主要有三点: (1) pid文件的内容:pid文件为文本文件,内容只有一行, 记录了该进程的ID. 用cat命令可以看到. (2) pid文件的作用:防止进程启动多个副本.只有获得pid文件(固定路径 ...

  3. 怎样让scratch里的人物两腿走动

    需要人物角色至少有两个“造型”,表现走路时的两个动作.以默认的“小猫”觉色为例,它有两个“造型”,可以用来表现奔跑的动作. 但是要想让小猫跑起来,需要脚本来实现,简单跑动脚本如下 scratch学习视 ...

  4. 微信小程序wx.setStorage(OBJECT)

    关于微信小程序的:wx.setStorage(OBJECT)在官网API介绍到:

  5. Java第十三天,内部类

    内部类 一.①成员内部类.②局部内部类(包含③匿名内部类) 1.内部类用外部类属性和方法的时候,可以随意进行访问. 2.外部类用内部类属性和方法的时候,需要通过内部类对象访问. 3.在编译成class ...

  6. 汇编刷题:求1000H单元开始的10个无符号字节数的最大值(本题放入了BL寄存器)

    DATA SEGMENT ORG 1000H INFO DB 1,2,3,4,5,70H,71H,72H,80H,92H MAX DB 00H DATA ENDS CODE SEGMENT ASSUM ...

  7. 跨域cookies 共享

    这是由于,本地调试.涉及到cookies的问题 想要跨域使用的问题 vue 中的mian.js中放入下面代码 import axios from 'axios' axios.defaults.with ...

  8. Three.js如何选中外部模型

    1.问题 three.js中模型选中使用的是射线法,根据摄像机角度,鼠标点击位置和模型选中的distance参数判断来选中模型.对于原生的矢量模型完全没有问题,但是当遇到导入的外部模型,如obj.st ...

  9. tensorflow2.x 报错 Could not load dynamic library 'cudart64_101.dll'

    当我们使用 tensorflow 最新版本的时候 ,会出现这样的错误 -- ::] Could not load dynamic library 'cudart64_101.dll'; dlerror ...

  10. jmeter事务控制器

    jmeter事务控制器常用于压力测试时如果一个功能包括多个请求时,需要测试这个功能的压力情况,则需要把多个请求放到一个事务控制器里面