SCAU 18216 银行服务

18216 银行服务

时间限制:1000MS  代码长度限制:10KB

题型: 编程题   语言: G++;GCC;VC

Description

银行通过叫号来决定服务用户的顺序,假设,银行有4类客户,分别用优先级1,2,3,4表示,级别高的优先得到服务。例如,当前有三个人排队,两个1级客户,一个3级客户,则银行叫号时,3级客户将先得到服务
,即使另两个1级的客户比他先到。当多个同级的客户将获得服务时,由先到的客户先得到服务。

假设,银行只有一个服务窗口,一次只能服务一个客户,假设该窗口每5分钟服务一个客户,即叫号的时刻分别为0分钟、5分钟、10分钟、……如果在叫号的时侯,没有客户,银行职员会去喝杯咖啡或上个洗手间,5分钟后再继续叫号。

有一种情况,银行工作到一定时间是要下班的,所以到一定时间,如果后面还有客户,将不再提供服务。

银行给出一系列客户到来的记录,每条记录包括“客户到来的时间”,“客户等级”,“客户姓名”(由一个单词构成),请输出该银行这一轮服务的客户的顺序。

输入格式

第一行两个数字,第一数字是客户的数量n(n<=100000),第二个数字是银行关门的时间,到这个时间,即关门,该点及之后,不再叫号,但之前已经叫号的客户会继续服务到完结。
此后,每一行是一个客户来访信息,包括3个数字,到来的时刻(分钟整点,最大10的8次方)、等级、姓名(最多20个字母)
(已经按到来时刻排序,注意:同一时刻可能会同时到来多个客户,这时若为同等级的,数据出现得早的稍早得到服务)

输出格式

按服务的先后顺序输出客户的姓名

输入样例

4 12
0 1 John
3 1 Smith
3 1 Tom
4 2 Flod

输出样例

John
Flod
Smith

问题分析:

本题可视为SCAU 18105 银行的叫号顺序加强版(增加下班时间限制)。

把每次叫号前到达的客户放进优先队列排序,即可确定本次叫号服务的客户。

不断重复直至下班。

(优先队列需重载排列方法)

代码实现:

#include<iostream>
#include<queue>
using namespace std;

//客户信息结构体
struct kehu
{
    int level;
    int ArriveTime;
    char name[25];
    int order;

    //自定义优先队列中客户信息排序方式
    bool operator<(const kehu& other)const
    {
        if(level!=other.level)                  //先比较等级
            return level<other.level;
        else if(ArriveTime!=other.ArriveTime)   //再比较到达时刻
            return ArriveTime>other .ArriveTime;
        else                                    //若等级、到达时刻均相同,则比较读入顺序
            return order>other.order;
    }
};

int main()
{
    //优先队列存储并排序客户信息
    priority_queue<kehu>line;
    int n=0,endtime=0,nexttime=0,ord=1;
    kehu temp;
    //nexttime:下一次叫号时间
    //endtime:下班时间

    cin>>n>>endtime;
    while(n--)
    {
        cin>>temp.ArriveTime>>temp.level>>temp.name;
        temp.order=ord++;

        here://跳转到这里,避免还未入队的temp被下一个客户覆盖

        //下班了
        if(nexttime>=endtime)
            return 0;

        //读入下一次叫号之前到达的客户
        if(temp.ArriveTime<=nexttime)
        {
            line.push(temp);
        }
        //下一次叫号前的客户已全部入队,则叫一次号
        else
        {
            if(!line.empty())
            {
                cout<<line.top().name<<endl;
                line.pop();
            }
            nexttime+=5;//注意下一次叫号时间+=5
            goto here;
        }
    }

    //没下班且还有客户在等待
    while(!line.empty()&&nexttime<endtime)
    {
        cout<<line.top().name<<endl;
        line.pop();
        nexttime+=5;
    }

    return 0;
}

  • 4
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值