问题描述
试题编号: | 202303-2 | |||||||||||||||||||||||||
试题名称: | 垦田计划 | |||||||||||||||||||||||||
时间限制: | 1.0s | |||||||||||||||||||||||||
内存限制: | 512.0MB | |||||||||||||||||||||||||
问题描述: | 问题描述顿顿总共选中了 n 块区域准备开垦田地,由于各块区域大小不一,开垦所需时间也不尽相同。据估算,其中第 i 块(1≤i≤n)区域的开垦耗时为 ti 天。这 n 块区域可以同时开垦,所以总耗时 tTotal 取决于耗时最长的区域,即:tTotal=max{t1,t2,⋯,tn} 为了加快开垦进度,顿顿准备在部分区域投入额外资源来缩短开垦时间。具体来说:
现在顿顿手中共有 m 单位资源可供使用,试计算开垦 n 块区域最少需要多少天? 输入格式从标准输入读入数据。 输入共 n+1 行。 输入的第一行包含空格分隔的三个正整数 n、m 和 k,分别表示待开垦的区域总数、顿顿手上的资源数量和每块区域的最少开垦天数。 接下来 n 行,每行包含空格分隔的两个正整数 ti 和 ci,分别表示第 i 块区域开垦耗时和将耗时缩短 1 天所需资源数量。 输出格式输出到标准输出。 输出一个整数,表示开垦 n 块区域的最少耗时。 样例输入1 Data 样例输出1 Data 样例解释如下表所示,投入 5 单位资源即可将总耗时缩短至 5 天。此时顿顿手中还剩余 4 单位资源,但无论如何安排,也无法使总耗时进一步缩短。
样例输入2 Data 样例输出2 Data 样例解释投入 20 单位资源,恰好可将所有区域开垦耗时均缩短为 k=2 天;受限于 k,剩余的 10 单位资源无法使耗时进一步缩短。 子任务70% 的测试数据满足:0<n,ti,ci≤100 且 0<m≤106; 全部的测试数据满足:0<n,ti,ci≤105 且 0<m≤109。 |
此题楼主最开始想复杂了,思路是将所有的天数和消耗按照一定的顺序排序,然后循环,模拟,利用两个指针改变来改变去,改变一次重新排序,但是这样的时间复杂度显然高得离谱
后来在csdn上翻找到了一个很好的方法【手把手刷CCF】202303-2-垦田计划100分(超简单思路,含详细解释注释与代码)-CSDN博客
由于不是我自己想的,此文章标注转载了哈
这个思路是用数组收容最大天数,然后创建一个数组以天数为下标记录相同天数的总消耗
按着他的思路我重新敲了一遍,最后70分显示运行错误,然后对cneed的容量进行了修改(最后很多的后项累加到前面面数字可能会很大)然后还是70分,显示错误,最后将cneed在全局进行声明(自动初始化)然后100过了。
代码如下:
#include<bits/stdc++.h>
using namespace std;
using ll=long long;
const int max_T=100010;
const int N=100010;
ll cneed[N];
//记录最大天数所需的总投入
int main(){
ll n,m,k;
cin>>n>>m>>k;
int t[n],c[n];
cin>>t[0]>>c[0];
int maxt=t[0];
cneed[t[0]]=c[0];
for(int i=1;i<n;i++){
scanf("%d %d",&t[i],&c[i]);
maxt=(maxt>t[i])?maxt:t[i];
cneed[t[i]]+=c[i];
}
for(int i=maxt;i>=k;i--){
if(cneed[i]<=m){//可以消耗一定的材料缩减时间
if(i==k){
cout<<k<<endl;
break;
}
else{
cneed[i-1]+=cneed[i];//使最大天数都减一天,更新新的最大天数所需的总材料
m-=cneed[i];
}
}
else{//无法消耗材料缩减一天
cout<<i<<endl;
break;
}
}
return 0;
}