CSP-202303-2-垦田计划

目录

一、题目描述

二、思路

三、C++实现如下


一、题目描述

问题描述

顿顿总共选中了 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 块区域的最少耗时。

二、思路

CCF第二题的套路感觉就是下手之前先想清楚哪部分能优化,现在的第二题感觉都是些递归的思想,题目要求我们把开垦时间缩短为最小天数,同时还要满足不能小于给定的最小天数,判断如何缩短当前最大天数,仅需知道如果要缩减这一天需要消耗多少的资源 ,与当前剩余资源比较即可得知能否缩减当天。那么我们进需要创建一个数组,将缩减某一天缩需要的资源数储存在其下表对应的数组中,接着从后往前依次遍历即可完成判断。

如何维护这个数组便是我们需要思考的问题,要缩减第i天,那么要在第i天消耗的资源数上加上第i+1天消耗的资源数量。而不用考虑小于i天的天数所消耗的资源。

那么我们就能得到dp[i] += dp[i+1]且从后向前遍历。

我的代码中将dp数组名命名为v,维护好了数组之后,我们再次从所需要的最大天数开始向前遍历dp数组,与当前剩余的资源数量比较即可完成判断。

三、C++实现如下

#include<iostream>
#include<vector>
#include<algorithm>
using namespace std;
//int v[200000];
vector<int> v(200000,0); 
int main(){
	int n,m,k,maxDay=0;
	cin>>n>>m>>k;
	
	for(int i = 0;i<n;++i){
		int t,c;
		cin>>t>>c;
		v[t] += c;
		maxDay = max(maxDay,t);
	}
	
	for(int i = maxDay;i>k;--i){
		v[i-1]+=v[i];
	}
//	for(int i = 1;i<=maxDay;++i){
//		cout << v[i] << ' ';
//	}
//	cout << endl;
	while(m>=v[maxDay]){
		m-=v[maxDay];
		maxDay-=1;
	}
	maxDay = max(maxDay,k); 
	cout << maxDay << endl;
	return 0;
} 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值