PAT甲级-1017 Queueing at Bank
题目链接
题意:
n个顾客在m个银行窗口排队,先来的顾客先服务,银行工作时间为早8点到晚17点,17点之后到的顾客不予服务,计算所有被服务的顾客的平均等待间。
要注意的点:
- 16:59:00来的顾客,处理时间为60分钟,银行在17点之后会继续服务直至服务完成,而不会准时关门。
- 16:58:00来排队的顾客,处理时间为60分钟,16:59:00来的顾客处理时间为60分钟。虽然第一个顾客服务完毕后已经超时了,但是第2个顾客也会服务。
- 换言之,只要在17:00:01这个时间来排队的顾客都会被服务,否则 Case 5测试点会WA。
题解:
先按照到达时间排序,每一次在窗口中选最小空闲时间的窗口给顾客服务,直到所有规定时间之前排队的顾客服务完。
#include<iostream>
#include<vector>
#include<algorithm>
using namespace std;
const int N=2e5;
int n,m;
struct node{
int arrive,process;
};
vector<node>customer;
int window[N];
bool cmp(node n1,node n2){
return n1.arrive>n2.arrive;
}
int sum,cnt;
int main(){
cin>>n>>m;
for(int i=0;i<n;i++){
int hour,minute,second,process;
scanf("%d:%d:%d %d",&hour,&minute,&second,&process);
int times=hour*60*60+minute*60+second;
customer.push_back({times,process});
}
for(int i=1;i<=m;i++){
window[i]=8*60*60;
}
sort(customer.begin(),customer.end(),cmp);
while(!customer.empty()){
int min_id,min_val=2e9;
for(int i=1;i<=m;i++){
if(window[i]<min_val){
min_val=window[i];
min_id=i;
}
}
// if(min_val>17*60*60)
// break;
node c=customer.back();
customer.pop_back();
if(c.arrive>17*60*60)
break;
cnt++;
sum+=max(0,min_val-c.arrive);
if(min_val<c.arrive){
window[min_id]+=c.arrive-min_val;
}
window[min_id]+=c.process*60;
}
if(cnt==0)
printf("%.1lf\n",0);
else
printf("%.1lf\n",sum*1.0/cnt/60);
return 0;
}