CCF-CSP202303-2 垦田计划 70分,进阶至100分过程

[问题描述]
顿顿总共选中了 n 块区域准备开垦田地,由于各块区域大小不一,开垦所需时间也不尽相同。据估算,其中第 i 块(1≤i≤n)区域的开垦耗时为 ti 天。这 n 块区域可以同时开垦,所以总耗时 tTotal 取决于耗时最长的区域,即:tTotal=max{t1,t2,⋯,tn}

为了加快开垦进度,顿顿准备在部分区域投入额外资源来缩短开垦时间。具体来说:

在第 i 块区域每投入 ci 单位资源,便可将其开垦耗时缩短 1 天;

耗时缩短天数以整数记,即第 i 块区域投入资源数量必须是 ci 的整数倍;

在第 i 块区域最多可投入 ci×(ti−k) 单位资源,将其开垦耗时缩短为 k 天;

这里的 k 表示开垦一块区域的最少天数,满足 0<k≤min{t1,t2,⋯,tn};换言之,如果无限制地投入资源,所有区域都可以用 k 天完成开垦。

现在顿顿手中共有 m 单位资源可供使用,试计算开垦 n 块区域最少需要多少天?

[输入格式]
从标准输入读入数据。

输入共 n+1 行。

输入的第一行包含空格分隔的三个正整数 n、m 和 k,分别表示待开垦的区域总数、顿顿手上的资源数量和每块区域的最少开垦天数。

接下来 n 行,每行包含空格分隔的两个正整数 ti 和 ci,分别表示第 i 块区域开垦耗时和将耗时缩短 1 天所需资源数量。

输出格式
输出到标准输出。

输出一个整数,表示开垦 n 块区域的最少耗时。

 【代码详解】

#include<iostream>
#include<algorithm>
using namespace std;
int n,m,k;
struct arr{
  int t;
  int c;	
}a[101];
bool cmp(arr&a,arr&b)
{
	return a.t>b.t;
}
int main()
{
  cin>>n>>m>>k;
  for(int i=0;i<n;i++)
  {
  	cin>>a[i].t>>a[i].c; 
  }
  sort(a,a+n,cmp);//将时间天数首先从大到小排列
  
  int range=0;
  int maxmin=a[0].t;//都先取那个最大的那个 
  int sum=0;
  for(;;)
  {
  if(maxmin==k)//当为2的时候说明已经之前计算到了2,不能往下执行会减少 
  break; 
  if(maxmin==a[range].t)
  {
  sum+=a[range].c;//加上耗时 ,注意sum要一直减,而不是重新赋值为0,
  //因为从调整的时候就会在一个水平线上,每次都要一起减。 
  range++;
  }
  else 
  {
  	if(m>=sum)
  	{
	  m=m-sum;
  	maxmin--;}
  	else
  	break;//已经超过了手上现有的资源 
  }	
  }cout<<maxmin;
  	
}

【做题感悟】

觉得逻辑思维要理得特别顺,特别是要注意点要将其从大到小排序,然后进行划分。

其中有特别多小点,比如==k是要记得跳出循环,一直减sum而不是要重复赋值为0,因为range过去后都在一条水平线上。

//分割线

后面又思考了一下,既然只有一个for循环,那时间复杂度消耗在了哪里,可以看到如果天数是一天天减的,那么在for循环中会达到O(m),那么想到阶段性考虑。

【100分代码及思路】

#include<iostream>
#include<algorithm>
using namespace std;
int n,m,k;
struct arr{
  int t;
  int c;	
}a[100010];
bool cmp(arr&a,arr&b)
{
	return a.t>b.t;
}
int main()
{
  cin>>n>>m>>k;
  for(int i=0;i<n;i++)
  {
  	cin>>a[i].t>>a[i].c; 
  }
  sort(a,a+n,cmp);//将时间天数首先从大到小排列
  
  int range=0;
  int maxmin=a[0].t;//都先取那个最大的那个 
  int sum=0;
  while(range<=n)//让时间复杂度变成O(n)
  {
  //不止可以一天天减 
  if(maxmin==a[range].t)
  {
  sum+=a[range].c;//加上耗时 ,注意sum要一直减,而不是重新赋值为0,
  //因为从调整的时候就会在一个水平线上,每次都要一起减。 
  range++;
  }
  else 
  {
  	int temp;
	  temp=(maxmin-a[range].t)*sum;//重要此处阶段性考虑
	  if(m>=temp)
	  {
	  	m=m-temp;
	  	maxmin=a[range].t;
	  } 
	  else
	  {
	  cout<<maxmin-m/sum;
	  return 0;//如果手上的资源不够 
      }
  }	
  }
  if((maxmin-m/sum)>k)//如果都弄完了还有资源就与最低值k作比较
  cout<<maxmin-m/sum; 
  else
  cout<<k; 
  
  return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值