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

转载请注明出处
作者:@ZJXin
由于本人水平有限,如有不正确的地方欢迎留言指出,谢谢!

考查Java的并发编程时,手写"生产者-消费者模型"是一个经典问题。本文给出一种相对比较简单容易理解、容易实现的java代码实现这一模型。

步骤:

  1. 创建一个实体类(本例为WoTou)
  2. 创建一个"容器"
  3. 创建生产者线程类
  4. 创建消费者线程类
  5. main()函数创建一个或多个生产者以及消费者线程,并同时运行

  1. 首先创建一个实体类

class WoTou{

    //用来标记生产的每个实例
    int id;
    
    public WoTou(int id){
        this.id = id;
    }
    
    public String toString(){
        //重写toString方法,方便打印
        return "WoTou:" + id;
    }
}
  1. 创建一个“容器”,用栈这一数据结构实现,生产者每次从将生产的物品放到最上边,同时消费者每次消费最上边的物品。(也可以用循环队列这一数据结构实现)
class Container{
    
    int index = 0;
    WoTou []arrWT = new WoTou[6];
    
    //通过synchronized这一关键字来同步
    public synchronized void set(WoTou wotou){//生产方法,加锁
        
        while(index == arrWT.length){
            //必须使用while,而不是if
            //因为如果用if抛出异常,后边的代码还会继续执行
            //while的话抛出异常后还是回到while里边执行
            try {
                this.wait();
            } catch ( InterruptedException e) {
                e.printStackTrace();
            }
        }
        
        //唤醒其他线程
        this.notifyAll();
        
        arrWT[index++] = wotou;
        
    }
    
    //通过synchronized这一关键字来同步
    public synchronized WoTou get(){//消费方法,加锁
        
        while(index == 0){
        //同理,必须使用while
            try{
                this.wait();
            }catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
        
        this.notifyAll();
        
        
        WoTou woTou = arrWT[--index];
        return woTou;
    }
    
}
  1. 创建生产者线程
class Producer implements Runnable{//生产者线程

    Container container = null;
    
    public Producer(Container container) {
        this.container = container;
    }
    
    @Override
    public void run() {
        
        for(int i=0;i<100;i++){
            
            WoTou wotou = new WoTou(i);
            container.set(wotou);
            System.out.println("生产了:" + wotou);
            try{
                Thread.sleep((int)(Math.random() * 200));
            }catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
    
}
  1. 创建消费者线程
class Consumer implements Runnable{//消费者线程

    Container container = null;
    
    public Consumer(Container container) {
        this.container = container;
    }
    
    @Override
    public void run() {

        while(true){
            
            WoTou woTou = container.get();
            System.out.println("消费了:"+woTou);
            
            try {
                Thread.sleep((int)(Math.random() * 1000));
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
    
}
  1. 主函数创建线程并运行
public class ProducerConsumerTest {

    /**
     * @param args
     */
    public static void main(String[] args) {
        Container container = new Container();
        Producer p = new Producer(container);
        Consumer c = new Consumer(container);
        new Thread(p).start();
        
        //可以创建多个生产者消费者线程
        //new Thread(p).start();
        //new Thread(p).start();
        //new Thread(c).start();
        new Thread(c).start();
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,969评论 19 139
  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,370评论 11 349
  • 看到一段话 “曹操再奸,都有知心朋友, 刘备再好,都有死对头, 孙权再温柔,两边都是仇。 不要太在乎别人对你的评价...
    瑾_174e阅读 251评论 2 2
  • 雨 携风 意阑珊 溪边竹坞 寻古书为伴 感悟良多 不知时 夜深 孤
    晏不知丶阅读 414评论 1 3
  • 有个人去乡村,途中遇到一条恶狗,慌乱之下因为害怕恶狗扑来,他立马做蹲下状,想吓唬狗。狗果然害怕逃跑了。 旁边一个老...
    小潘在等你阅读 993评论 3 8