AcWing 11. 背包问题求方案数(01背包计数)

简介: 笔记

背包问题求方案数


题意

有 N 件物品和一个容量是 V 的背包。每件物品只能使用一次。


第 i 件物品的体积是 vi,价值是 wi。


求解将哪些物品装入背包,可使这些物品的总体积不超过背包容量,且总价值最大。


输出 最优选法的方案数。注意答案可能很大,请输出答案模 109+7 的结果。


思路

问题背景时 01背包问题 所以求价值的最大值可以直接用 01背包的思路求解:19.png


代码

#include <bits/stdc++.h>
//#define int long long
#define INF 0x3f3f3f3f
#define mod 1000000007
#define endl '\n'
#define rep(i, st, ed) for (int (i) = (st); (i) <= (ed);++(i))
#define pre(i, ed, st) for (int (i) = (ed); i >= (st);--(i))
using namespace std;
typedef long long LL;
typedef pair<int, int> PII;
inline int gcd(int a, int b) { return b ? gcd(b, a % b) : a; }
inline int lowbit(int x) { return x & -x; }
const int N = 1200;
int n, m;
int w[N], v[N];
int f[N];
int g[N]; // 价值为f[i] 的方案数 
void solve() {
  cin >> n >> m;
  for (int i = 1; i <= n; ++i)cin >> w[i] >> v[i];
  // 价值为f[0] 即价值为0时的方案数为 1 那就是都不选 
  g[0] = 1;
  // 01背包
  for (int i = 1; i <= n; ++i) {
    for (int j = m; j >= w[i]; --j) {
      int maxn = max(f[j], f[j - w[i]] + v[i]);
      int cnt = 0;
      if (maxn == f[j])cnt += g[j];
      if (maxn == f[j - w[i]] + v[i])cnt += g[j - w[i]];
      f[j] = maxn;
      g[j] = cnt % mod;
    }
  }
  // 求出价值最大为多少
  int cnt = 0;
  for (int i = 0; i <= m; ++i) {
    cnt = max(cnt, f[i]);
  }
  //枚举体积 如果当前体积下价值与最大价值相等  那么答案加上 g[j]
  int res = 0;
  for (int i = 0; i <= m; ++i) {
    if (f[i] == cnt)
      res = (res + g[i]) % mod;
  }
  cout << res << endl;
}
signed main() {
  // int t;cin >> t;
  // while(t--)
  solve();
  return 0;
}


目录
相关文章
|
3月前
acwing 789 数的范围
acwing 789 数的范围
27 4
|
7月前
|
存储
【洛谷 P1255】数楼梯 题解(递归+记忆化搜索+高精度)
这是一个使用动态规划解决的“数楼梯”问题。给定楼梯数`N`,求不同上楼方式的数量。程序通过递归函数`f()`计算,其中`f(x) = f(x - 1) + f(x - 2)`,初始条件`f(1) = 1`,`f(2) = 2`。考虑到数据规模可能很大,使用了高精度加法运算。样例输入`4`,输出`5`。代码中定义了一个存储中间结果的向量`mem`,并提供了一个用于显示高精度数的`printv()`函数。
122 0
|
5月前
|
算法 Java
LeetCode初级算法题:子数组最大平均数+二叉树的最小深度+最长连续递增序列+柠檬水找零
LeetCode初级算法题:子数组最大平均数+二叉树的最小深度+最长连续递增序列+柠檬水找零
47 0
|
7月前
|
存储 JavaScript 算法
【背包问题】01背包问题和完全背包问题的模板
【背包问题】01背包问题和完全背包问题的模板
|
人工智能 算法 决策智能
动态规划之背包问题(01背包问题、完全背包问题、方案数填满型背包问题)
动态规划之背包问题(01背包问题、完全背包问题、方案数填满型背包问题)
298 1
|
测试技术 容器
动态规划之背包问题——背包三讲(01背包,完全背包,多重背包)(三)
动态规划之背包问题——背包三讲(01背包,完全背包,多重背包)(三)
238 0
动态规划之背包问题——背包三讲(01背包,完全背包,多重背包)(三)
|
C++
(排列,选择类dp)(数论同余定理,同余运算)(以背包为母题)1214. 波动数列
(排列,选择类dp)(数论同余定理,同余运算)(以背包为母题)1214. 波动数列
102 0
|
算法 C++
【基础算法】几种特殊数(素数、公约数、完全数、亲密数) & C++实现
素数又称为质数,它指在一个大于1的自然数中,除了1和它自身外,没法被其他自然数整除的数。比1大,但不是素数的数称为合数。0和1既不是素数,也不是合数。因为素数的分布没有明显的规律,所以在程序中一般根据素数的定义来判断该数是否为素数。例如哥德巴赫猜想:哥德巴赫通过大量的数据猜测,所有不小于6的偶数,都可以表示为两个奇素数之和。后人将其称之为“1+1”。并且,对于每个不小于9的奇数,都可以表示为三个奇素数之和。
351 0
【基础算法】几种特殊数(素数、公约数、完全数、亲密数) & C++实现
|
JavaScript 测试技术 vr&ar
动态规划之背包问题——背包三讲(01背包,完全背包,多重背包)(一)
动态规划之背包问题——背包三讲(01背包,完全背包,多重背包)(一)
162 0
动态规划之背包问题——背包三讲(01背包,完全背包,多重背包)(一)
|
测试技术
动态规划之背包问题——背包三讲(01背包,完全背包,多重背包)(二)
动态规划之背包问题——背包三讲(01背包,完全背包,多重背包)(二)
300 0
动态规划之背包问题——背包三讲(01背包,完全背包,多重背包)(二)