7-2 拼题A打卡奖励 (25 分) 特殊的01背包

拼题 A 的教超搞打卡活动,指定了 N 张打卡卷,第 i 张打卡卷需要 mi​ 分钟做完,完成后可获得 ci​ 枚奖励的金币。活动规定每张打卡卷最多只能做一次,并且不允许提前交卷。活动总时长为 M 分钟。请你算出最多可以赢得多少枚金币?

输入格式:

输入首先在第一行中给出两个正整数 N(≤103) 和 M(≤365×24×60),分别对应打卡卷的数量和以“分钟”为单位的活动总时长(不超过一年)。随后一行给出 N 张打卡卷要花费的时间 mi​(≤600),最后一行给出 N 张打卡卷对应的奖励金币数量 ci​(≤30)。上述均为正整数,一行内的数字以空格分隔。

输出格式:

在一行中输出最多可以赢得的金币数量。

输入样例:

5 110
70 10 20 50 60
28 1 6 18 22

输出样例:

40

样例解释:

选择最后两张卷子,可以在 50+60=110 分钟内获得 18+22=40 枚金币。

思路:(dp自上而下)

显然这是一个01背包问题。但是,按照常规的想法,把时间看成是物品的体积,金币看成是价值,则 M 就是背包的容量。但是根据题目的数据范围, M(≤365×24×60)M(≤365×24×60) ,复杂度 NM 肯定会TLE。

我们再看一下题目给定其他数据范围,显然 ci<30 成为题目转化的一个突破口。按照原来做法就是用最小容量求最大价值,现在反过来用最大价值求最小容量。最大价值:1000∗30, 复杂度:1000∗30∗1000 。

#include
#include
using namespace std;
const int M = 365*24*60, N = 1010;
int a[N],w[N];
int dp[30011];// i金币 dp[i]时间 
int main(){memset(dp,0x3f,sizeof(dp));dp[0] = 0;int n,m,sum = 0;cin>>n>>m; for(int i = 1; i <= n; i++) cin>>a[i];for(int i = 1; i <= n; i++) {cin>>w[i];sum += w[i];}for(int i = 1; i <= n; i++){for(int j = sum; j >= w[i]; j--){// dp自上而下if(dp[j] > dp[j-w[i]] + a[i]){dp[j] = dp[j-w[i]] + a[i];}}}for(int i = sum; i >= 0; i--){if(dp[i] <= m) {cout<


本文来自互联网用户投稿,文章观点仅代表作者本人,不代表本站立场,不承担相关法律责任。如若转载,请注明出处。 如若内容造成侵权/违法违规/事实不符,请点击【内容举报】进行投诉反馈!

相关文章

立即
投稿

微信公众账号

微信扫一扫加关注

返回
顶部