题意:
有n个订单和可以在m小时内制作月饼,制作月饼不考虑时间(即,你可以在一个时刻在所有需要的月饼都做完)
接下来是n个订单的信息:需要在mon月,d日,year年,h小时交付订单r个月饼
接下来一行t,s表示制作的月饼可以保质t小时,每保质一小时需要花费s的价值
接下来m行表示从第0小时开始在该时间制作月饼的花费的价值(2000年1月1日0时表示第0个小时)
求完成所有订单消耗的最小价值
题解:
因为题目中给的时间是年月日的,所以我们首先要把它转化为距离2000年1月1日0时多少小时
然后又因为我们需要求最小花费,对于一个订单,你可以在该订单结束的时刻(设为距离2000年1月1日0时x小时)的区间[0,x]内中选择花费最少的一天去制作月饼。
这就可以用到单调队列,构造一个从左向右递增的序列即可。这样的话对于x时刻的月饼,我们只需要找到单调队列中那个最左边满足题意得就可以了
如果这样的话,那么这个单调队列按照从左向右递增的这个值就要考虑到保质花费,和在那个时刻做月饼得花费。具体见代码吧
代码:
1 #include <stdio.h> 2 #include <string.h> 3 #include <stdlib.h> 4 #include <iostream> 5 #include <algorithm> 6 #include<map> 7 using namespace std; 8 typedef long long LL; 9 typedef unsigned long long ull; 10 const int inf=0x3f3f3f3f; 11 const LL INF=0x3f3f3f3f3f3f3f3fll; 12 const int maxn=1e5+5; 13 map<string,int>mp; 14 int n,m,time[3050],sum[15],S,T,cost[100050],q[100050]; 15 LL num[3000]; 16 void init() 17 { 18 mp["Jan"]=1,mp["Feb"]=2,mp["Mar"]=3,mp["Apr"]=4,mp["May"]=5,mp["Jun"]=6,mp["Jul"]=7,mp["Aug"]=8; 19 mp["Sep"]=9,mp["Oct"]=10,mp["Nov"]=11,mp["Dec"]=12; 20 sum[0]=0,sum[1]=31,sum[2]=sum[1]+28,sum[3]=sum[2]+31,sum[4]=sum[3]+30,sum[5]=sum[4]+31,sum[6]=sum[5]+30; 21 sum[7]=sum[6]+31,sum[8]=sum[7]+31,sum[9]=sum[8]+30,sum[10]=sum[9]+31,sum[11]=sum[10]+30; 22 } 23 24 int get(int y,int m,int d,int t) //获取这个时间对应的2000年1月1日0点有多少小时 25 { 26 int ans=0; 27 for(int i=2000; i<y; i++) 28 { 29 if((i%4==0&&i%100!=0)||i%400==0) 30 ans+=366; 31 else 32 ans+=365; 33 } 34 if((y%4==0&&y%100!=0)||y%400==0) 35 { 36 ans+=sum[m-1]; 37 if(m-1>=2)ans++; 38 } 39 else 40 { 41 ans+=sum[m-1]; 42 } 43 ans+=(d-1); 44 return ans*24+t; 45 } 46 47 int main() 48 { 49 init(); 50 while(~scanf("%d%d",&n,&m)) 51 { 52 if(n==0&&m==0)break; 53 for(int i=0; i<n; i++) 54 { 55 int year,day,t; 56 char mon[10]; 57 scanf("%s%d%d%d%I64d",mon,&day,&year,&t,&num[i]); 58 time[i]=get(year,mp[mon],day,t); 59 //printf("->%d\n",time[i]); 60 } 61 scanf("%d%d",&T,&S); 62 int tail=0,head=0,k=0; 63 LL cnt=0; 64 for(int i=0; i<m; i++) 65 { 66 scanf("%d",&cost[i]); 67 /* 68 维护一个递增的序列,这个递增的值是花费多少才能使月饼保存到第i个小时 69 */ 70 while(head<tail&&cost[q[tail-1]]+S*(i-q[tail-1])>=cost[i])tail--; 71 q[tail++]=i; 72 while(i==time[k]) //当这个条件触发时,就可以从单调队列中拿出来你从0时刻到现在那个最小的花费 73 { //但是还要判断一下保鲜时间超时了没有 74 while(head+1<tail&&(i-q[head]>T))head++; 75 cnt+=num[k]*(cost[q[head]]+S*(i-q[head])); 76 k++; 77 } 78 } 79 printf("%I64d\n",cnt); 80 } 81 return 0; 82 }