看程序员怎么解决食堂排队问题

点击上方“程序人生”,选择“置顶公众号”

[转载出处:www.ii77.com]

第一时间关注程序猿(媛)身边的故事

[好文分享:www.ii77.com]


看程序员怎么解决食堂排队问题

作者

寒食君i

原文标题

食堂中的生产-消费模型

如需转载,请联系原作者授权。


在学校的时候,我不爱去食堂成功,一是由于暗黑料理,更重要的一点是人太多了,队伍往往从窗口排到了门口,点菜、计算价格、付款三种业务由打饭阿姨一人完成,思维切换忙碌,操作变更频繁,导致效率低下,降低了食堂的吞吐量,造成了不好的用户体验。

看程序员怎么解决食堂排队问题

而最近在公司食堂吃饭,发现是另外一种设计:工作人员向餐台上添加食物,两条通道对顾客进行分流,顾客依次进入队列然后对餐台上的食物进行自取,最后在队列出口进行结账。

变成了这样:

看程序员怎么解决食堂排队问题

在学校食堂提供的打饭服务中,是由打饭阿姨单独完成一系列的操作。而在公司食堂,则将打饭解耦成了排盘与收银两个服务,降低了每个工作人员的工作复杂度,使其能更加专注于自己负责的工作。

同时,在学校食堂中,一名学生在打饭的同时,排在队伍后面的同学是出于等待状态,等待获取打饭阿姨的服务,这对于打饭阿姨的性能是极大的浪费。

而在公司食堂,这是一种生产-消费的模型。我们来看看都有些什么?

首先,消费者是谁?肯定是来用餐的顾客,消费的东西是什么?是餐台上的食物。而生产者则是将食物摆上餐台的工作人员。

那么将这个场景抽象化,可以用这张图来表示:

看程序员怎么解决食堂排队问题

这张图主要包含了这些信息:

  • 一个数据仓库

    用于数据缓存,生产者生产完数据放入,消费者从中取出。它有两个注意点:

    1. 同步,即生产者和消费者不能同时访问数据仓库。对应到现实世界,你可以理解为:顾客和工作人员不会在一个瞬间去访问餐台,两个顾客也不能在同一瞬间去争抢同一盘食物。

    2. 当仓库满了,生产者无法再添加数据进去,将处于阻塞状态,并提醒消费者进行消费;当仓库为空,消费者无法从中获取数据。处于阻塞状态,并提醒生产者进行生产。


  • 两种角色:生产者、消费者。

  • 两种关系:

    1. 依赖关系:生产者与消费者

    2. 竞争关系:生产者与生产者、消费者与消费者

那如何使用这种模型,将食堂这个场景来实现还原呢?我们可以这么分析:

  1. 数据仓库可以使用一个队列或数组来实现。但是要注意同步。我们这里选择java.util.concurrent包下的BlockingQueue,它有着已经实现的阻塞添加和阻塞删除的特性。当然也可以用一个List去自己实现。

  2. 生产者是一些线程,生产一些数据,将其入队。

  3. 消费者是一些线程,从队首获取数据,将其消费。

核心概念就是以上三点,那么我们接下来用代码简单实现一下。预期目标是让整个流程自动运行起来。

首先我们来定义一个食物类Food,这里从简,只给它一个id属性。

 1public class Food {
2    private int id;
3    //含参构造函数,这里用不到无参构造器,所以可以不写。否则新建对象将会出错。
4    public Food(int id){
5        this.id=id;
6    }
7    //get/set
8    public int getId() {
9        return id;
10    }
11    public void setId(int id) {
12        this.id = id;
13    }
14}

接着我们来编写生产者类Producer和消费者Consumer,让我们来想想这两个类该怎么去编写?

首先我们需要使用组合的方式来加入一条BlockingQueue(阻塞队列),它是整个问题的核心,又由于这个容器是只用来存放Food的,我们可以加上泛型。

之前也提到了,Food有一个属性是id,那么这个id从何而来呢?应该是生产者生产Food时赋予的,而由于多位生产者在同时生产,为了保证id的唯一性,我们需要对其进行原子化的自增操作。这里我们可以使用java.util.concurrent.atomic包中的AtomicInteger,当然如果使用int,然后自己做一些同步操作也是可以的。(若这里对Java内存模型JMM不了解的,可以去阅读我写过的这篇文章:由浅入深Java内存模型

那么,Producer类可以这么写,将每个工作线程当作一名生产者,并作一些必要的文字输出,方便控制台观察:

 1public class Producer implements Runnable {
2    private boolean working=true;
3    private BlockingQueue queue;
4    private static AtomicInteger count = new AtomicInteger();
5    //构造函数
6    public Producer(BlockingQueue queue){
7        this.queue=queue;
8    }
9    @Override
10    public void run() 
{
11        while(working){
12            int id=count.incrementAndGet();
13            Food food=new Food(id);
14//            System.out.println(Thread.currentThread().getId()+"号员工开始工作");
15            if(queue.offer(food)){
16                System.out.println(Thread.currentThread().getId()+"号员工将"+food.getId()+"号食物加入餐台");
17            }else {
18                System.out.println("餐台已满,"+food.getId()+"号食物无法加入");
19            }
20            try {
21                Thread.sleep(1000*3);
22            } catch (InterruptedException e) {
23                e.printStackTrace();
24            }
25        }
26    }
27    public  void stop(){
28        working=false;
29    }
30}

同理,Consumer可以这么写:

 1public class Consumer implements Runnable {
2    private boolean working=true;
3    private BlockingQueue queue;
4//含参构造函数
5    public Consumer(BlockingQueue queue){
6        this.queue=queue;
7    }
8    @Override
9    public void run() 
{
10        while (true){
11            try {
12                Food food=queue.take();//take()方式,若队列中没有元素则线程被阻塞
13                System.out.println(food.getId()+"号食物已被"+Thread.currentThread().getId()+"号顾客端走");
14                Thread.sleep(1000*2);
15            } catch (InterruptedException e) {
16                e.printStackTrace();
17            }
18        }
19
20    }
21}

最后,就是主函数调用了,我们使用线程池来对线程进行分配,这里我将数据队列定义为10个元素空间,线程池使用了newFixedThreadPool方式来规定5条线程,初始化3名生产者和15名消费者。

Main.java

 1public class Main {
2    public static void main(String[] args) {
3        BlockingQueue queue = new LinkedBlockingDeque<>(10);
4        Producer[] p=new Producer[3];
5        Consumer[] c=new Consumer[15];
6        for (int i=0;i<3;i++){
7            p[i]=new Producer(queue);
8        }
9        for (int j=0;j<15;j++){
10            c[j]=new Consumer(queue);
11        }
12        ExecutorService executorService= Executors.newFixedThreadPool(5);
13        for (int i=0;i<3;i++){
14            executorService.execute(p[i]);
15        }
16        for (int j=0;j<15;j++){
17            executorService.execute(c[j]);
18        }
19        try {
20            Thread.sleep(1000*20);
21        } catch (InterruptedException e) {
22            e.printStackTrace();
23        }
24    }
25}

下面就是食堂开张的时刻了

热门文章

  1. 汉语拼音字母歌儿歌(汉语拼音字母歌26个儿歌视频)2024-05-12
  2. 闺蜜群名字好听点的(闺蜜群名字有创意)2024-05-12
  3. 广联达马凳筋怎么设置(广联达马凳筋参数设置)2024-05-12
  4. 2024届浙江省金华十校高三下学期4月模拟考试语文试卷2024-05-12
  5. fif英语测试系统(fif英语测试如何作弊)2024-05-12
  6. 农村户口迁城市户口的利与弊2021(农村户口贷款需要什么条件)2024-05-12
  7. 手机投影到电视最简单(用手机投影到电视上)2024-05-12
  8. 尿毒症长期透析副作用大?要警惕这6大并发症,一个比一个严重2024-05-12
  9. 人教版八年级上册数学知识点归纳图文(人教版八年级上册数学知识点归纳(精编版))2024-05-12
  10. 侠盗飞车罪恶都市长夜(侠盗飞车罪恶都市长夜)2024-05-12
  11. 剪切的快捷键是什么(键盘上剪切的快捷键是什么)2024-05-12
  12. 前列腺癌内分泌治疗出现不良反应怎么办?2024-05-12
  13. 玫瑰花的葬礼背后的故事(玫瑰花的葬礼背后的故事是什么)2024-05-12
  14. 头层牛皮和二层牛皮哪个好(皮鞋材质头层牛皮和二层牛皮哪个好)2024-05-12
  15. 糠秕孢子菌(糠秕孢子菌怎么引起的)2024-05-12
  16. 深圳福田盛钱包pos机代理商(深圳福田盛钱包pos机代理商)2024-05-12
  17. 蜉蝣是什么(蜉蝣是什么动物)2024-05-12
  18. 谥(谥法解)2024-05-12
  19. 产品型号是什么(什么叫产品型号)2024-05-12
  20. 类风湿关节炎患者如何调养及食谱2024-05-12
自媒体 微信号:ii77 扫描二维码关注公众号
爱八卦,爱爆料。

小编推荐

  1. 1 打脸?公有云巨头纷纷进军私有领域究竟是何故?

    不知不觉,公有云巨头们在私有云中也掀起了不小的波澜。

  2. 2 晒晒新家!108㎡清新北欧风,配色好好看,最爱客餐厅!

    108㎡简约北欧风,配色十分清新,各种软装也搭配得甚是好看。

  3. 3 65㎡温馨浪漫的北欧风 设计很独特

    小户型总是给人一种很温暖的感觉,没有空荡荡的,自己的小窝,每个小细节都能让你想起温馨的事情。65㎡的户型

  4. 4 谢娜、应采儿被老公宠成女皇,为何她却成了郭晓冬的“女奴”?

    这样的爱情会幸福吗?

  5. 5 头条|万华化学:从MDI龙头变身为全球综合化工企业

    2018年6月4日,万华化学复牌,标志着万华化学从2017年12月5日起停牌,并用6个月的时间筹划的整体上市

  6. 6 群星闪耀共赴2018群星MARS创新创业大赛

    距离2018群星MARS创业大赛发布会倒计时1天!8月30日,北京德必WE国际文化创意中心,由群星资本、多

  7. 7 美,可以成为生产力;同时也要提升你的审美力

    到底“美”了,可以干什么用!

  8. 8 这里的水泥砖创意,别的地方找不到

    只是用混凝土砖建房子未免太没创意了做一个有创意的人让无趣的生活变得更有意义比如只是用来建造围墙让创意改变生活

  9. 9 品牌取名指南:这样取名字,能帮你减少50%以上的广告费用

    读前思考3秒,你觉得怎样才是好的品牌名字?本文暂时抛开品牌品质、经营、规模等问题,重点和你聊聊品牌取名的话题。

  10. 10 豆瓣、猫眼和 IMDb 的电影评分是怎么得出来的?

    若批评无自由则赞美无意义

本文内容来自网友供稿,如有信息侵犯了您的权益,请联系反馈核实

Copyright 2024.爱妻自媒体,让大家了解更多图文资讯!