You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

75 lines
3.1 KiB

2 years ago
#include <bits/stdc++.h>
using namespace std;
const int N = 110, M = N << 1;
int v[N], w[N];
int n, m, root;
// f[u][i][j] 以u为根在前i个子树中(组)选择最大体积不超过j的所有方案
// 属性max价值
int f[N][N][N];
// 链式前向星
int e[M], h[N], idx, ne[M];
void add(int a, int b) {
e[idx] = b, ne[idx] = h[a], h[a] = idx++;
}
// u:以u为根的树
// 返回值以u为根的树一级子树son的个数
int dfs(int u) {
/*
u,j>=v[u], uw[u]
uu
*/
for (int j = v[u]; j <= m; j++) f[u][0][j] = w[u];
// ② 考虑u的每个子树
int s = 0;
for (int i = h[u]; ~i; i = ne[i]) { // 枚举u的每个子节点
s++; // 前s个子树
int son = e[i]; // 在链式前向星中,这是几号节点
int c = dfs(
son); // 对子节点son,把它的f[son][c][k]信息填充完整,返回son子树的一级结点个数
/*
:f[u][i][j],
i:
1~dfs(u)dfs(u)u0,
j: uj
kjson,k
[0,j-v[u]],sonf[son][c][k] f[u][s -
1][j -
k]:ss-1sonkf[u][s-1][j-k]
*/
for (int j = v[u]; j <= m; j++)
for (int k = 0; k <= j - v[u]; k++)
f[u][s][j] = max(f[u][s][j], f[u][s - 1][j - k] + f[son][c][k]);
}
return s; // 返回u有多少个子结点
}
int main() {
// 初始化链式前向星
memset(h, -1, sizeof h);
cin >> n >> m; // 物品个数和背包容量
for (int i = 1; i <= n; i++) { // n个物品
int p;
cin >> v[i] >> w[i] >> p; // 体积、价值、父亲
if (p == -1)
root = i; // 对一棵树而言,根是最重要的
else
add(p, i); // 从父亲向儿子建边,一般树都是从上到下建边
}
int s = dfs(root); // 从根节点开始遍历填充DP Table,返回值是root有几个子树
/*
Q:dfsroot
A:DPf[u][i][j]uij,
f[root][root][m]
root
*/
printf("%d\n", f[root][s][m]);
return 0;
}