luogu
树形DP从下往上把状态更新,既然需要从叶结点更新到根节点,所以需要先从根节点先深搜到叶结点,然后才能从树下端更新到树上端。
三道模板很开心。
P1352
题意:舞会开始,一个人的上司来了,他就肯定不来........有向边 终点存在的话 那起点一定不存在, 求最多多少个点
#include <math.h>
#include <algorithm>
#include <vector>
#include <queue>
using namespace std;
const int maxn = 6010;
int n;
int u, v;
int r[maxn];
vector<int> son[maxn];
int dp[maxn][2];
int vis[maxn];
void Dp(int x) {
dp[x][0] = 0;
dp[x][1] = r[x];
int len = son[x].size();
for (int i = 0; i < len; i++) {
Dp(son[x][i]);
dp[x][0] += max(dp[son[x][i]][0], dp[son[x][i]][1]);
dp[x][1] += dp[son[x][i]][0];
}
}
int main () {
scanf("%d", &n);
for (int i = 1; i <= n; i++) {
scanf("%d", &r[i]);
}
while (scanf("%d%d", &u, &v) && (u || v)) {
son[v].push_back(u);
vis[u] = 1;
}
int rt;
for (int i = 1; i <= n; i++) {
if (vis[i] == 0) {
rt = i;
break;
}
}
// printf("rt = %d\n", rt);
Dp(rt);
printf("%d\n", max(dp[rt][1], dp[rt][0]));
return 0;
}
知识兔P2016
和第一个差不多,不一样的地方在于双向边,边的起点和终点可以同时存在,求的是最少可以放多少个点
#include <algorithm>
#include <math.h>
using namespace std;
const int maxn = 1510;
const int inf = 1e9+7;
struct node {
int v, nxt;
}edge[maxn * maxn];
int head[maxn], tot;
void Insert(int u, int v) {
edge[++tot].v = v;
edge[tot].nxt = head[u];
head[u] = tot;
}
int n, m, u, v;
int dp[maxn][2];
void dfs(int u, int fa) {
dp[u][0] = 0;
dp[u][1] = 1;
for (int i = head[u]; i; i = edge[i].nxt) {
// printf("u=%d fa=%d\n", u, fa);
int v = edge[i].v;
if (v == fa) continue;
dfs(v, u);
dp[u][0] += dp[v][1];
dp[u][1] += min(dp[v][0], dp[v][1]);
}
}
int main () {
scanf("%d", &n);
for (int i = 0; i < n; i++) {
scanf("%d%d", &u, &m);
for (int j = 0; j < m; j++) {
scanf("%d", &v);
Insert(u, v);
Insert(v, u);
}
}
dfs(0, -1);
printf("%d\n", min(dp[0][0], dp[0][1]));
return 0;
}
知识兔P2015
树形dp + 01背包
#include <algorithm>
#include <cstring>
#include <math.h>
using namespace std;
const int maxn = 105;
const int maxm = 3e4 + 10;
const int inf = 1e9+7;
struct node {
int v, nxt, w;
}edge[maxn * 2];
int head[maxn], tot;
void Insert(int u, int v, int w) {
edge[++tot].v = v;
edge[tot].w = w;
edge[tot].nxt = head[u];
head[u] = tot;
}
int n, m, u, v, w, q;
int dp[maxn][maxn];
void dfs(int u, int fa) {
for (int i = head[u]; i; i = edge[i].nxt) {
int v = edge[i].v, w = edge[i].w;
if (v == fa) continue;
dp[v][1] = w;
dfs(v, u);
for (int j = q; j; j--) {//开始背包了,子树的最优解现在可以看成是新的物品的价值,这里整体的感觉给我像是把诸多二叉树的状态整合成了一根链状的dp[v]的状态
for (int k = 0; k <= j; k++) {
if ((k != j && j != 1) || u == 1)//并没有搞明白这里是怎么一回事
dp[u][j] = max(dp[u][j], dp[u][j-k] + dp[v][k]);
}
}
}
}
int main () {
scanf("%d%d", &n, &q);
for (int i = 1; i < n; i++) {
scanf("%d%d%d", &u, &v, &w);
Insert(u,v,w);
Insert(v,u,w);
}
dfs(1, 0);
printf("%d\n", dp[1][q]);
return 0;
}
知识兔