问题描述
顿顿总共选中了 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 块区域的最少耗时。
样例输入1(本题我主要基于样例1来进行详细解释)
4 9 2
6 1
5 1
6 2
7 1
样例输出1
5
样例解释
如下表所示,投入 5 单位资源即可将总耗时缩短至 5 天。此时顿顿手中还剩余 4 单位资源,但无论如何安排,也无法使总耗时进一步缩短。
i | 基础耗时 ti | 缩减 1 天所需资源 ci | 投入资源数量 | 实际耗时 |
---|---|---|---|---|
1 | 6 | 1 | 1 | 5 |
2 | 5 | 1 | 0 | 5 |
3 | 6 | 2 | 2 | 5 |
4 | 7 | 1 | 2 | 5 |
代码及注释(满分易懂无算法):
#include<iostream>
#include<algorithm>
using namespace std;
const int maxn=1e5+10;
long long n,m,k;
long long date[maxn]={0};
int main()
{
cin>>n>>m>>k;
int t,c,maxt;
for(int i=0;i<n;i++)
{
cin>>t>>c;
date[t]+=c;//记录耗时为t的开垦时间去减少1所需要花费的资源数目,date[6]被合并了,共同需要3资源,因为我们只考虑最大的t,所以可以将其合并,优先考虑date[7]去减少1个时间的资源,然后9-1=8,资源剩下8,date[6]就变成1+2+1=4,然后8-4=4;接下来date[5]就变成1+1+2+1=4+5本身资源为1=5,需要5个资源,所以不够了,所以就到5这里结束了
maxt=max(maxt,t);
}
int flag=k;
for(int i=maxt;i>k;i--)
{
if(m>=date[i])//begin:date[7]=1 next:date[6]=4 end:date[5]=5
{ //m=8 m=4
date[i-1]+=date[i];//date[6]=1+2+1=4,m=9-1=8
m-=date[i];//date[5]=date[6]+date[5]=4+1=5,m=8-date[6]=8-4=4
//在上一轮m=4,而目前date[5]=5,不满足,跳出
}
else
{
flag=i;
break;
}
}
cout<<flag<<endl;
return 0;
}