题解 | #【模板】完全背包#

【模板】完全背包

https://www.nowcoder.com/practice/deda4293d9b24ce1aeaf1813c88b8c25

题目链接

【模板】完全背包

题目描述

给定 种物品和一个容量为 的背包。第 种物品的体积是 ,价值是 。每种物品都有无限件可用。求解将哪些物品装入背包,可使这些物品的总体积不超过背包容量,且总价值最大。

输入:

  • 第一行输入一个整数 ,表示有 组测试数据。
  • 对于每组测试数据:
    • 第一行输入两个整数 ,分别表示物品数量和背包容量。
    • 接下来 行,每行输入两个整数 ,分别表示第 件物品的体积和价值。

输出:

  • 对于每组测试数据,输出一个整数,表示能获得的最大总价值。

解题思路

本题是典型的 完全背包问题 模型。与01背包问题不同的是,每种物品可以选取无限次。我们可以使用动态规划来解决。

  1. 状态定义:

    • 我们定义一个一维数组 ,其中 表示背包容量为 时,可以获得的最大总价值。
  2. 状态转移:

    • 对于第 种物品(体积为 ,价值为 ),我们可以选择不放或者放一个或多个。
    • 状态转移方程为:
    • 这个方程的含义是,对于当前容量 ,我们可以选择不放入第 种物品,此时最大价值就是 (由前 种物品决定);或者选择放入第 种物品,此时背包的剩余容量为 ,最大价值为 。我们在这两种选择中取最大值。
  3. 遍历顺序:

    • 遍历物品是外层循环,遍历背包容量是内层循环。
    • 与01背包的关键区别在于,内层循环(背包容量)必须是正序遍历(从 )。
    • 为什么要正序遍历?因为 在计算 时,应该是已经考虑过第 种物品之后的值。这样就相当于允许第 种物品被重复选取。如果逆序遍历,那么在计算 时, 还是只考虑了前 种物品的状态,这就变成了01背包问题。
  4. 初始化:

    • 我们需要将 数组全部初始化为0,因为在不放任何物品时,价值为0。

最终, 就是我们要求的答案。

代码

#include <iostream>
#include <vector>
#include <algorithm>

using namespace std;

int main() {
    int T;
    cin >> T;
    while (T--) {
        int N, V;
        cin >> N >> V;
        vector<int> v(N), w(N);
        for (int i = 0; i < N; ++i) {
            cin >> v[i] >> w[i];
        }

        vector<long long> dp(V + 1, 0);

        for (int i = 0; i < N; ++i) {
            for (int j = v[i]; j <= V; ++j) {
                dp[j] = max(dp[j], dp[j - v[i]] + w[i]);
            }
        }
        cout << dp[V] << "\n";
    }
    return 0;
}
import java.util.Scanner;

public class Main {
    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        int T = sc.nextInt();
        while (T-- > 0) {
            int N = sc.nextInt();
            int V = sc.nextInt();
            int[] v = new int[N];
            int[] w = new int[N];
            for (int i = 0; i < N; i++) {
                v[i] = sc.nextInt();
                w[i] = sc.nextInt();
            }

            long[] dp = new long[V + 1];

            for (int i = 0; i < N; i++) {
                for (int j = v[i]; j <= V; j++) {
                    dp[j] = Math.max(dp[j], dp[j - v[i]] + w[i]);
                }
            }
            System.out.println(dp[V]);
        }
    }
}
T = int(input())
for _ in range(T):
    N, V = map(int, input().split())
    items = []
    for i in range(N):
        v, w = map(int, input().split())
        items.append((v, w))

    dp = [0] * (V + 1)

    for v, w in items:
        for j in range(v, V + 1):
            dp[j] = max(dp[j], dp[j - v] + w)

    print(dp[V])

算法及复杂度

  • 算法:动态规划(完全背包)
  • 时间复杂度: - 其中 是测试数据组数, 是物品数量, 是背包容量。
  • 空间复杂度: - 主要开销是 数组。
全部评论

相关推荐

当年还在美团那个倒霉的&nbsp;Peppr&nbsp;团队工作时,我一直有个疑问:这群人每天到底在自嗨什么。每次开会一堆人围着一堆“看起来很高级”的文档转,模板统一、名词复杂、页数感人,每一页都在暗示一件事:“你不懂,是因为你不专业。”但现实是——代码照样写在&nbsp;💩&nbsp;山上,该出问题还是会出问题,这真的很逗,系统一出问题,文档的唯一作用就是证明:“我们当初确实认真写过文档。”所以本质区别到底是什么?是代码质量提升了,还是大家在精神层面完成了一次“工程师&nbsp;cosplay”?有句话说得好潮水退去才知道谁在裸泳。还记得当时的马哥、明哥(图&nbsp;1&nbsp;左)最爱反复强调一句话:“所有场景一定要想到。”、“这个场景为什么没考虑到?”不过他们这些话我是真的听进去了。不然我也不会在一年多前就说:这个项目活不过两年。顺带一提,那段时间还有个固定节目。每次下楼,总能听见我明哥在吐槽不同的人。我从他身后绕过去,经常能听到他一边抽烟一边说:“xx&nbsp;这小子太坑了,回头我一定要跟马哥说说。”于是深谙人情世故但真不会抽烟的我也会从口袋掏出一支低尼古丁含量的烟给自己点上,假意自己什么都没听到什么都不知道,只是来抽烟的。后来我才明白,这可能也是团队文化的一部分:问题永远在别人身上,而我们,永远在复盘里😂。
秋招白月光
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务