P1048 [NOIP2005 普及组] 采药

[NOIP2005]采药

辰辰是个天资聪颖的孩子,他的梦想是成为世界上最伟大的医师。为此,他想拜附近最有威望的医师为师。医师为了判断他的资质,给他出了一个难题。医师把他带到一个到处都是草药的山洞里对他说:"孩子,这个山洞里有一些不同的草药(共 m 株),采每一株都需要一些时间,每一株也有它自身的价值。我会给你一段时间,在这段时间里,你可以采到一些草药。如果你是一个聪明的孩子,你应该可以让采到的草药的总价值最大。"

现在已知山洞中共有 n 株草药,采第 i 株采药需要花费 t[i] 时间,可以获得 c[i] 价值。

采药的时间 T(1≤T≤1000) ,一共有不超过 100 个草药,每个草药的价值和消耗的时间不超100 。

我们可以看出这是一道 01 背包 的题目,可以设 f[j] 表示消耗时间为 j 时可以获得的最大价值和,然后进行通过递推进行转移:

cpp 复制代码
for (int i = 1; i <= n; i++) {
    for (int j = T; j >= t[i]; j--) {
        f[j] = max(f[j], f[j - t[i]] + c[i]);
    }
}
cout << f[T] << endl;

实际上我们还可以通过dfs求解小数据时的答案。设

cpp 复制代码
void dfs(int x, int sumt, int sumc)

表示在前 1∼(x−1) 株药草中,已经花费了 sumt 的时间,采了 sumc 价值的药。

cpp 复制代码
int n, T;
int dfs(int x, int sumt) {
    if (x > n) return 0;
    int r = 0;
    if (sumt + t[x] <= T) {
        r = dfs(x + 1, sumt + t[x]) + c[x];
    }
    r = max(r, dfs(x + 1, sumt));
    return r;
}

cout << dfs(1, 0);

我们尝试用记忆化搜索的方式来求解这个动态规划方程。

我们用 f[i][j] 表示在 i,i+1,⋯,n−1,n 中选择物品,剩余时间为 j 时能选出的最大价值和。

然后 DFS 时每次枚举某一个物品选择或者不选,当某些状态已经得到答案时,可以直接返回结果。

cpp 复制代码
int n, T, f[101][1001];
int dfs(int x, int sumt) {
    if (x > n) return 0;
    if (f[x][sumt] != -1) return f[x][sumt];
    int r = 0;
    if (sumt + t[x] <= T) {
        r = dfs(x + 1, sumt + t[x]) + c[x];
    }
    r = max(r, dfs(x + 1, sumt));
    return f[x][sumt] = r;
}

cout << dfs(1, 0);
完整代码
cpp 复制代码
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
int t[101], c[101], n, f[101][1001], T; // t[i] 为耗费的时间, c[i] 为价值
int dfs(int x, int sumt) {
    if (x > n) return 0;
    if (f[x][sumt] != -1) return f[x][sumt];
    int r = 0;
    if (sumt + t[x] <= T) {
        r = dfs(x + 1, sumt + t[x]) + c[x];
    }
    r = max(r, dfs(x + 1, sumt));
    return f[x][sumt] = r;
}

int main() {
    scanf("%d%d", &T, &n);
    for (int i = 1; i <= n; i++) {
        scanf("%d%d", &t[i], &c[i]);
    }
    memset(f, -1, sizeof(f)); // 将所有状态初始化为未计算
    printf("%d\n", dfs(1, 0));
    return 0;
}
相关推荐
charlee444 分钟前
将std容器的正向迭代器转换成反向迭代器
c++
CoovallyAIHub15 分钟前
为什么85%的企业AI项目都失败了?
深度学习·算法·计算机视觉
KarrySmile18 分钟前
Day8--滑动窗口与双指针--1004. 最大连续1的个数 III,1658. 将 x 减到 0 的最小操作数,3641. 最长半重复子数组
数据结构·算法·双指针·滑动窗口·不定长滑动窗口·最大连续1的个数·最长子数组
arbboter18 分钟前
【C++20】新特性探秘:提升现代C++开发效率的利器
c++·c++20·新特性·span·结构化绑定·初始化变量·模板参数推导
zc.ovo21 分钟前
图论水题4
c++·算法·图论
KyollBM27 分钟前
【Luogu】每日一题——Day20. P4366 [Code+#4] 最短路 (图论)
算法·图论
qqxhb28 分钟前
零基础数据结构与算法——第七章:算法实践与工程应用-金融算法
算法·风险评估算法·金融算法·交易策略算法·欺诈检测算法
眠りたいです1 小时前
Qt音频播放器项目实践:文件过滤、元数据提取与动态歌词显示实现
c++·qt·ui·音视频·媒体·qt5·mime
墩墩同学1 小时前
【LeetCode题解】LeetCode 74. 搜索二维矩阵
算法·leetcode·二分查找
SunnyKriSmile1 小时前
输入10个数并求最大值
c语言·算法