一扶苏一
2019-08-05 15:12:21
给定一棵
首先看一个trick:
考虑如下遍历一棵树的伪代码:
func dfs(u):
size[u] <- 1
for v in clild[u] do
dfs(v)
for i = 1 : size[u] do
for j = 1: size[v] do
do sth
end
end
size[u] <- size[u] + size[v]
end
end func
如果认为do sth的复杂度是
证明上可以考虑如果记录每个点的 dfs 序,
回到这个题,最直接的DP是设
考虑将上述状态更换为以
具体的,设
那么转移显然:
其中
例如子树中选择了
其中
考虑上述转移中,
再转移时需要保证
在考虑最后一个问题:我们设计的方程是填表转移,即枚举了上面伪代码中
func dfs(u):
size[u] <- 1
for v in clild[u] do
dfs(v)
size[u] <- size[u] + size[v]
for sum = 1 : size[u] do
for i = max(0, sum - size[v]) : size[v] do
do sth
end
end
end
end func
虽然写成这样的时间复杂度十分难以分析,但是枚举和再枚举其中一个的复杂度显然和枚举两个求和的复杂度相同,因此上述代码的时间复杂度也为
由于单次转移是
#include <cstdio>
#include <cstring>
#include <algorithm>
const int maxn = 2003;
int n, K, dK;
int sz[maxn];
ll frog[maxn][maxn];
struct Edge {
int v, w;
Edge *nxt;
Edge(const int _v, const int _w, Edge *h) : v(_v), w(_w), nxt(h) {}
};
Edge *hd[maxn];
void dfs(const int u, const int fa);
int main() {
freopen("1.in", "r", stdin);
qr(n); qr(K); dK = n - K;
for (int i = 1, u, v, w; i < n; ++i) {
u = v = w = 0; qr(u); qr(v); qr(w);
hd[u] = new Edge(v, w, hd[u]);
hd[v] = new Edge(u, w, hd[v]);
}
dfs(1, 0);
qw(frog[1][K], '\n', true);
return 0;
}
void dfs(const int u, const int fa) {
sz[u] = 1;
memset(frog[u] + 2, -1, 16008);
for (auto e = hd[u]; e; e = e->nxt) if (e->v != fa) {
int v = e->v; dfs(v, u); sz[u] += sz[v];
for (int i = std::min(sz[u], K); ~i; --i) {
for (int j = i, lim = std::max(0, i - sz[v]); j >= lim; --j) if (~frog[u][j]) {
int k = i - j; if (frog[v][k] == -1) continue;
ll val = (1ll * k * (K - k) + 1ll * (sz[v] - k) * (dK - sz[v] + k)) * e->w;
frog[u][i] = std::max(frog[u][i], frog[u][j] + frog[v][k] + val);
}
}
}
}
1、上面那种神奇的枚举方式遍历整棵树的时间复杂度是
2、当问题不满足最优子结构时,可以考虑DP每个子问题对答案的贡献。
3、很多时候刷表法难以滚动数组,需要转化成填表法
感谢 @DDOSvoid 大爷与我的讨论以及对我的启发