29.JAVA编程之生产者与消费者案例

生产者与消费者

生产者不断生产产品,消费者不断取走产品
实例:饭店里有一个厨师和一个服务员,这个服务员必须等待厨师准备好膳食,当厨师准备好时,他会通知服务员上来,如何返回继续等待,这是一个任务协作的示例,厨师代表生产者,而服务员代表消费者
示例代码:

//两个线程协同工作,先生产再消费的过程
public class ThreadDemo2 {
    public static void main(String[] args) {
        Food f = new Food();
        Producter p = new Producter(f);
        Customers c = new Customers(f);
        Thread t1 = new Thread(p);
        Thread t2 = new Thread(c);
        t1.start();
        t2.start();
    }
}

//先调用set才能调用get
//生产者对象
class Producter implements Runnable{
    private Food food;
    //通过构造器把food传进来

    public Producter(Food food){
        this.food = food;
    }
    @Override
    public void run() {
        //模拟生产20份菜
        for (int i = 0; i < 20; i++) {
            if(i%2==0){
                food.set("火锅","海底捞");
            }else
            {
                food.set("佛跳墙","贵得很");
            }
        }
    }
}

//消费者
class Customers implements Runnable{
    private Food food;
    public Customers(Food food){
        this.food = food;
    }
    @Override
    public void run() {
        for (int i = 0; i < 20; i++) {
            food.get();
        }
    }
}
//食物
class Food{
    private String name;
    private String desc;
    private boolean flag = true;//true表示可以生产,flase表示可以消费
    //生产产品
    public synchronized void set(String name,String desc){
        //不能生产时
        if (!flag){
            try {
                this.wait(); //线程进入等待状态,释放监视器的所有权(对象锁),直到notify唤醒
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
            this.setName(name);
            try {
                Thread.sleep(500);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            this.setDesc(desc);
            flag = false;
            this.notify();  //唤醒等待的线程(随机的其中一个)
        }

    //消费产品
    public synchronized void get(){
       //不能消费
        if (flag){
            try {
                this.wait();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
            try {
                Thread.sleep(500);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }

            System.out.println(this.getName()+"->"+this.getDesc());
            flag = true;
            this.notifyAll();
    }

    public void setName(String name) {
        this.name = name;
    }

    public void setDesc(String desc) {
        this.desc = desc;
    }

    public String getName() {
        return name;
    }

    public String getDesc() {
        return desc;
    }

    @Override
    public String toString() {
        return super.toString();
    }

    public void Food(String name,String desc){
        this.name = name;
        this.desc = desc;
    }

    public void Food(){

    }
}

运行效果:

sleep与wait的区别:
sleep:让线程进入休眠状态,让出CPU的时间片,但是不释放对象监视器的所有权(对象锁)
wait:让线程进入等待状态,让出CPU的时间片,并释放对象监视器的所有权,等待其他线程通过notify方法来唤醒

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 本文出自 Eddy Wiki ,转载请注明出处:http://eddy.wiki/interview-java.h...
    eddy_wiki阅读 2,248评论 0 14
  • 本文主要讲了java中多线程的使用方法、线程同步、线程数据传递、线程状态及相应的一些线程函数用法、概述等。 首先讲...
    李欣阳阅读 2,503评论 1 15
  • Java多线程学习 [-] 一扩展javalangThread类 二实现javalangRunnable接口 三T...
    影驰阅读 2,994评论 1 18
  • 林炳文Evankaka原创作品。转载自http://blog.csdn.net/evankaka 本文主要讲了ja...
    ccq_inori阅读 671评论 0 4
  • 一、进程和线程 进程 进程就是一个执行中的程序实例,每个进程都有自己独立的一块内存空间,一个进程中可以有多个线程。...
    阿敏其人阅读 2,626评论 0 13