背包问题求方案数
题意
有 N 件物品和一个容量是 V 的背包。每件物品只能使用一次。
第 i 件物品的体积是 vi,价值是 wi。
求解将哪些物品装入背包,可使这些物品的总体积不超过背包容量,且总价值最大。
输出 最优选法的方案数。注意答案可能很大,请输出答案模 109+7 的结果。
思路
问题背景时 01背包问题 所以求价值的最大值可以直接用 01背包的思路求解:
f [ j ] = m a x ( f [ j ] , f [ j − w [ i ] ] + v [ i ] ) f[j] = max(f[j],f[j - w[i]] + v[i]) f[j]=max(f[j],f[j−w[i]]+v[i])
设 g [ j ] g[j] g[j] 表示价值为 f [ j ] f[j] f[j] 时的方案数
那么有如下转移方程
-
如果 f [ j ] < f [ j − w [ i ] ] + v [ i ] f[j] < f[j - w[i]] + v[i] f[j]<f[j−w[i]]+v[i]
g [ j ] = g [ j − w [ i ] ] g[j] = g[j - w[i]] g[j]=g[j−w[i]]
-
如果 f [ j ] > f [ j − w [ i ] ] + v [ i ] f[j] > f[j - w[i]] + v[i] f[j]>f[j−w[i]]+v[i]
g [ j ] = g [ j ] g[j] = g[j] g[j]=g[j]
-
如果 f [ j ] = = f [ j − w [ i ] ] + v [ i ] f[j] == f[j - w[i]] + v[i] f[j]==f[j−w[i]]+v[i]
g [ j ] = g [ j ] + g [ j − w [ i ] ] g[j] = g[j] + g[j - w[i]] g[j]=g[j]+g[j−w[i]]
代码
#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;
}

本文介绍了一种解决背包问题中寻找最大价值及其方案数量的方法。通过01背包问题的思路,利用动态规划求解最优价值及对应方案数,并考虑答案过大时采用取模操作。
&spm=1001.2101.3001.5002&articleId=116139041&d=1&t=3&u=27ca83c9b667484093a3db9fe4b8e189)
1257

被折叠的 条评论
为什么被折叠?



