#10009. 「一本通 1.1 练习 5」钓鱼

1 篇文章 0 订阅

关于这一道题,真的是一个悲催,由于本宫太菜了,所以打一遍就忘一遍

于是,打这一遍的时候,我就决定要写一篇题解来纪念一下~~~

题目描述

在一条水平路边,有 nn 个钓鱼湖,从左到右编号为 1,2,…,n1,2,…,n。佳佳有 HH 个小时的空余时间,他希望利用这个时间钓到更多的鱼。他从 11 出发,向右走,有选择的在一些湖边停留一定的时间(是 55 分钟的倍数)钓鱼。最后在某一个湖边结束钓鱼。佳佳从第 ii 个湖到第 i+1i+1 个湖需要走 5\times T_i5×Ti​ 分钟路,还测出在第 ii 个湖停留,第一个 55 分钟可以钓到 F_iFi​ 条鱼,以后每再钓 55 分钟,可以钓到的鱼量减少 D_iDi​,若减少后的鱼量小于 00,则减少后的鱼量为 00 。为了简化问题,佳佳假定没有其他人钓鱼,也没有其他因素影响他钓到期望数量的鱼。请编程求出佳佳最多能钓鱼的数量。

输入格式

第一行一个整数 nn,表示湖的个数

第二行一个整数 HH,表示佳佳的空闲时间

第三行有 nn 个整数,依次表示每个湖第一个 55 分钟能钓到鱼的数量

第四行有 nn 个整数,依次表示以后的每5分钟钓鱼数量比前一个 55 分钟钓鱼数量减少的数量

第五行有 n-1n−1 个整数,T_iTi​ 表示由第 ii 个湖到第 i+1i+1 个湖需要花 5\times T_i5×Ti​ 分钟的路程

输出格式

输出只有一行,表示佳佳最多能钓鱼的数量。

样例

样例输入

3
1
4 5 6
1 2 1
1 2

样例输出

35

样例解释

在第 11 个湖钓 1515 分钟,共钓得 4+3+2=94+3+2=9 条鱼;

在第 22 个湖钓 1010 分钟,共钓得 5+3=85+3=8条鱼;

在第 33 个湖钓 2020 分钟,共钓得 6+5+4+3=186+5+4+3=18 条鱼;

从第 11 个湖到第 22 个湖,从第 22 个湖到第 33 个湖,共用时间 1515 分钟,共得 3535 条鱼,并且这是最多的数量。

数据范围与提示

对于 100\%100% 的数据,2\le n\le 100, 1\le H\le 202≤n≤100,1≤H≤20。

 

解题思路

这道题在loj上的标记是贪心,但是个人感觉貌似用dp也许会更好理解一点(注意是理解,不是更优)

对于这一道题,我定义了一个二维的dp

dp[i][j]表示在前i个鱼塘(此时已在第i个鱼塘)钓了j分钟的鱼(j是不包含两两鱼塘之间的路程时间的

这道题的难点就是路程时间进行了一定的干扰,所以在方程转移的时候直接把这一个干扰项排除就行了(从起始点到每个鱼塘的路程是一定的)

然后三重for就搞定了

第一重表示:在第几个鱼塘

第二重表示:已经钓了多少分钟

            第三重表示:在当前鱼塘钓了多久的鱼

 

代码呈现

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
int n,h,a[210],b[210],c[210],maxx=-0xfffffff;
int dp[2100][2100];
int main()
{
	scanf("%d%d",&n,&h);h*=12;
	for(int i=1;i<=n;i++)scanf("%d",&a[i]);
	for(int i=1;i<=n;i++)scanf("%d",&b[i]);
	c[1]=0;
	for(int i=1;i<n;i++)
	{
		int x;scanf("%d",&x);
		c[i+1]=c[i]+x;
	}
	memset(dp,0,sizeof(dp));
	for(int i=1;i<=n;i++)
	{
		for(int j=1;j<=h-c[i];j++)
		{
			dp[i][j]=dp[i][j-1];
			for(int k=0;k<=j;k++)
			{
				if(a[i]<(k-1)*b[i])break;
				int sum=0;for(int z=0;z<k;z++)sum+=a[i]-(b[i]*z);
				//sum=k*(a[i]+a[i]-(k-1)*b[i])/2;(sum既可以用for求,也可以用等差数列~~)
				dp[i][j]=max(dp[i][j],dp[i-1][j-k]+sum);
			}
			maxx=max(maxx,dp[i][j]);
		}
	}
	printf("%d\n",maxx);
	return 0;
}

 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值