java多线程之消费者生产者模式

Wesley13
• 阅读 530
/*@author shijin
* 生产者与消费者模型中,要保证以下几点:
* 1 同一时间内只能有一个生产者生产        生产方法加锁sychronized
* 2 同一时间内只能有一个消费者消费        消费方法加锁sychronized
* 3 生产者生产的同时消费者不能消费        生产方法加锁sychronized
* 4 消费者消费的同时生产者不能生产        消费方法加锁sychronized
* 5 共享空间空时消费者不能继续消费        消费前循环判断是否为空,空的话将该线程wait,释放锁允许其他同步方法执行
* 6 共享空间满时生产者不能继续生产        生产前循环判断是否为满,满的话将该线程wait,释放锁允许其他同步方法执行   
*/

//主类
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;
    }

    /** 
    * show 生产进程. 
    */ 
    public void run(){
        for(int i = 0;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(){
        for(int i = 0;i < 20;i++){
            Mantou m = ss.pop();
//            System.out.println("消费了:---------" + m + " 共" + ss.index + "个馒头");
//    同上    在上面一行进行测试也是不妥的,对index的访问应该在原子操作里,因为可能在pop之后此输出之前又生产了,会产生输出混乱
            try{
                Thread.sleep((int)(Math.random()*1000));
            }catch(InterruptedException e){
                e.printStackTrace();
            }
        }
    }
}
点赞
收藏
评论区
推荐文章
待兔 待兔
6个月前
手写Java HashMap源码
HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程22
捉虫大师 捉虫大师
3年前
一种极致性能的缓冲队列
本文已收录https://github.com/lkxiaolou/lkxiaolou欢迎star。背景在多线程下的生产者消费者模型中,需求满足如下情况:对生产者生产投递数据的性能要求非常高多个生产者,单个(多个也可以,本文只介绍单个的情况)消费者当消费者跟不上生产者速度时,可容忍少部分数据丢失生产者是单条单条地生产数据举个日志采集的例子,日志在不同的
九章 九章
3年前
一 java线程的等待/通知模型
java中线程之间的通信问题,有这么一个模型:一个线程修改了一个对象的值,而另一个线程感知到了变化,然后进行相应的操作,整个过程开始于一个线程,而最终执行又是另一个线程。前者是生产者,后者就是消费者,也可以叫做生产者消费者问题生产者生产了产品,如何通知消费者?下面就介绍下java线程中的等待通知机制。其它语言类似,自行研究。代码附上下面是以买小
Stella981 Stella981
3年前
Linux系统 Centos7 环境基于Docker部署Rocketmq服务
消息队列基本概述MQ,MessageQueue,基于TCP协议构建的简单协议,区别于具体的通信协议。基于通信协议定义和抽象的更高层次的通信模型,一般都是生产者和消费者模型,又或者说服务端和客户端模型。生产者/消费者模型:一般通过定义生产者和消费者实现消息通信从而屏
Wesley13 Wesley13
3年前
3.rabbitmq
rabbitmq发布订阅模式模型组成一个消费者Producer,一个交换机Exchange,多个消息队列Queue,多个消费者Consumer一个生产者,多个消费者,每一个消费者都有自己的一个队列,生产者没有将消息直接发送到队列,而是发送到了交换机,每个队列绑定交换机,生产者发送
Wesley13 Wesley13
3年前
ActiveMQ消息队列详解
activemq消息队列,分为生产者和消费者。.1创建生产者publicclassProducter{//ActiveMq的默认用户名privatestaticfinalStringUSERNAMEActiveMQConnection.DEFAULT\_USER;//ActiveMq的默认登录密码pr
Stella981 Stella981
3年前
Qt中的QThread:使用QSemaphore进行多线程数据同步
20210127:在生产者、消费者的方法中添加线程挂起方法QThread::usleep(10),使ui不卡。20210128:在添加Track类(保存生产者Producer生成的每组数据),在ui界面中使用modelview同步显示生产者生成的数据,modelview不会对主线程造成卡顿。对消费者同样创建view,还没有进行model绑定。避免
Wesley13 Wesley13
3年前
Java中生产者与消费者模式
 生产者消费者模式首先来了解什么是生产者消费者模式。该模式也称有限缓冲问题(英语:Boundedbufferproblem),是一个多线程同步问题的经典案例。该问题描述了两个共享固定大小缓冲区的线程——即所谓的“生产者”和“消费者”——在实际运行时会发生的问题。生产者的主要作用是生成一定量的数据放到缓冲区中,然后重复此过程。与此同
Wesley13 Wesley13
3年前
Java多线程之线程协作
常见的线程协作方式是:生产者/消费者。一个线程作为生产者,生产要处理数据,比如拿一个线程来生产Order,用户每下一单,此线程就生产一个Order对象。设置一个仓库,来存放生产出来的Order对象。一个线程作为消费者,消费|处理仓库中的Order对象(打印订单、拣货、发货)。demo  订单处理流程1、用
Easter79 Easter79
3年前
SpringCloud 系列文章
SpringCloud生产者与消费者上一篇文章我们介绍了Euarka的搭建,本篇文章,我们搭建俩个服务,生产者服务与消费者服务。我们就以电商系统为例:服务生产者,订单查询服务orderserver,服务消费者orderclient说明:orderserver服务提供查询订单信息的