2018-12-25 线程生产消费者模式

原理: 建立一个公共的容器用来存放产品(产品池),生产者和消费者分别为单独的线程对产品池进行操作。为了保证线程的安全需要利用synchronized来使得产品池来实现阻塞的功能(互锁)。

建立产品池(容器):

public class Container<T> {                       //建立产品池的泛型
    private T[] arr;                              //泛型数组
    private  int top=0;                          //数组指针,指示数组的元素序号          
    public Container() {                        //确定容器的容量
        this(5);
    }

    public Container(int size) {                    //容器的构造方法
        this.arr = (T[])new Object[size];
    }

    public synchronized void save(T o){              //生产者生产产品的同步方法
        while (top==arr.length){                     //判定数组内没有元素
            try {
                this.wait();                          //等待
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
}
        arr[top]=o;                               //添加元素到数组
        for (int i=0;i<arr.length;i++)
        {
         top++;                                    //指针移动
        System.out.println("做了"+top+"个包子");
}
        notify();                                  //通知等待的对象
        System.out.println("有包子了");
    }

测试:

public class Test {
    public static void main(String[] args) {     //主线程
        Container<String> container = new Container<>(); //创建容器对象
        new Thread("做包子") {                      //创建生产者的线程(匿名类)
            @Override
            public void run() {
                for (int i = 0; i < 10; i++)         //生产者生产出的产品数量
                    container.save("包子");
                System.out.println("最后存量:" + container.count());
            }
        }.start();
        new Thread("取包子1") {   //消费者1
            @Override
            public void run() {
                for (int i = 0; i < 100; i++) //保持有消费者进行购买
                    container.fetch();
            }
        }.start();
        new Thread("取包子2") {
            @Override
            public void run() {
                container.fetch();//消费者购买一次
            }
        }.start();
    }
}

此功能实现了在最大存货量达成后,在有消费者需求将不再回应,此时消费者保持等待(过程结束)。这类方法是利用了数组建立栈模型来存放产品,实际中利用队列的模型(或者是)更合适。

为了使主程序简洁,可以单独提出产品类、消费者类、生产者类使得整个程序结构更加优化

产品类:

public class BaoZi {
    private Integer id;
    private String name;

    public BaoZi() {

    }

    public BaoZi(Integer id, String name) {
        this.id = id;
        this.name = name;
    }

    public Integer getId() {
        return id;
    }

    public void setId(Integer id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

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

    @Override
    public String toString() {
        return "BaoZi{" +
                "id=" + id +
                ", name='" + name + '\'' +
                '}';
    }
}

生产者类:

public class Productor extends Thread{
    private Container<BaoZi> container;
    private Integer sum;

    public Productor(Container<BaoZi> container,Integer sum,String name) {
        super(name);
        this.container=container;
        this.sum = sum;
    }

    @Override
    public void run() {
        String[] names={"鲜肉包","香菇包","豆沙包"};
        Random random=new Random();
        for(int i=0;i<sum;i++){
             BaoZi baozi=new BaoZi(i+1,names[random.nextInt(3)]);
            System.out.println(Thread.currentThread().getName()+"做一个"+baozi);
            container.save(baozi);
        }
    }
}

消费者类:

public class Customer extends Thread{
    private Container<BaoZi>container;
    private Integer sum;

    public Customer(Container<BaoZi>container,Integer sum,String name) {
        super(name);
        this.container=container;
        this.sum = sum;
    }

    @Override
    public void run() {

        Random random=new Random();
        for(int i=0;i<sum;i++){
            BaoZi baoZi=container.fetch();
            System.out.println(Thread.currentThread().getName()+"买了一个"+baoZi);
            try {
                Thread.sleep(1000+random.nextInt(1000));
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

产品池:

public class Container<T> {
    private T[] arr;
    private int top = 0;
    public Container() {
        this(5);
    }

    public Container(int size) {
        this.arr = (T[]) new Object[size];
    }

    public synchronized void save(T o) {
        while (top == arr.length) {
            try {
                this.wait();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
}
      arr[top] = o;
          top++;
     notify();
  }
public synchronized T fetch() {
        while (top == 0) {
            try {
                this.wait();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
        try {
            Thread.sleep(400);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        top--;
        T t = arr[top];
        notify();
         return t;
    }
 public int count() {
        return top;
    }
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,194评论 6 490
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,058评论 2 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 156,780评论 0 346
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,388评论 1 283
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,430评论 5 384
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,764评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,907评论 3 406
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,679评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,122评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,459评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,605评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,270评论 4 329
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,867评论 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,734评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,961评论 1 265
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,297评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,472评论 2 348

推荐阅读更多精彩内容