题目链接:https://ac.nowcoder.com/acm/contest/11257/F
大意:
n块肉饼,m个平底锅。每块肉饼有其烹饪时间,最多只能先后在两个平底锅中。
问:最少用多少时间能将所有肉饼烹饪,按时间输出每块肉饼的烹饪方案。
思考:
注意:肉饼放到两个平底锅时,时间要错开。所以不能简单的抽象成物品放到盒子中。
先确定所用的最少总时间:
所用的最少总时间(每口锅烹饪的肉饼最长时间)为:max(肉饼中的最长烹饪时间,每个锅的平均时间)
-
如果
肉饼中的最长烹饪时间
比每个锅的平均时间(总烹饪时间/锅的数量)
大,那必定要用一个锅单独烹饪这块肉饼。反证:假如说
每口锅的最长时间点
为平均时间
,那有最长烹饪时间的这块饼占满了这口锅的时间,还要转移到其他锅中,那转移过再烹饪的时间要和当前锅的时间错开,也就是要延续这段时间,这就超过最长时间点(平均时间)了,矛盾。 -
如果
肉饼中的最长烹饪时间
比平均时间
小,那这块肉饼可以直接在一个锅 x x x中烹饪完,或者烹饪到这口锅的最长时间后,转移到下一口锅 y y y,时间从0开始,恰好补上上口锅开始烹饪该块之前的时间。
相当于先在锅y中烹饪一段时间,再转移到锅 x x x中。
确定了最小总时间,剩下的就只要将每块肉饼放到锅中就行了。
如果当前锅能放下,就放下;放不下,就放到下一个锅中。
实现:
遍历每块肉饼,如果当前肉饼烹饪时间+该锅已经消耗的时间
不超过锅的最长烹饪时间
,说明当前肉饼能直接在该锅全部烹饪;
如果超过了锅的最长烹饪时间
,就要转移到下一个锅中,从时间0开始,到剩余的时间。注意时间顺序。
Code:
#define ll long long
#include<iostream>
using namespace std;
const int N=200010;
int n,m,k,T,cnt,flag;
int ans,a[N];
ll sum,maxa;
int main(){
cin>>n>>m;
for(int i=1;i<=n;i++) cin>>a[i],sum+=a[i],maxa=max(maxa,a[i]*1ll);
ll t=sum/m;
if(sum%m) t++;
maxa=max(maxa,t); //最小总时间(每口锅的最长烹饪时间)
int pos=1; //锅的位置
t=0; //该锅消耗的时间
for(int i=1;i<=n;i++)
{
if(t+a[i]<=maxa){ //能放下
cout<<1<<" ";
cout<<pos<<" "<<t<<" "<<t+a[i]<<endl;
t+=a[i];
if(t==maxa){ //用完了该锅的时间
t=0;
pos++;
}
}
else //当前锅烹饪不完第i块肉饼
{
cout<<2<<" ";
cout<<pos+1<<" "<<0<<" "<<a[i]-(maxa-t)<<" "; //剩下的时间;注意排在前面
cout<<pos<<" "<<t<<" "<<maxa<<endl;
t=a[i]-(maxa-t);
pos++;
}
}
return 0;
}
关键是注意到,时间是错开的!然后求出最小总时间。
贪心地将每个物品放进来。。
有点秒啊。。