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;
}