Hamburger Steak(思维,贪心)

29 篇文章 2 订阅
8 篇文章 1 订阅

题目链接: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;
}

关键是注意到,时间是错开的!然后求出最小总时间。
贪心地将每个物品放进来。。
有点秒啊。。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值