【题解】蓝魔法师
蓝魔法师
https://ac.nowcoder.com/acm/problem/20811
题意:有一棵
个点的树,求出有多少种断边方案,使得每个连通块大小不超过
。
。
边连接的是树上的父亲和儿子节点,可以直接通过树边转移。因此考虑以  为根变成有根树,然后进行树上 DP。
记  表示以 
 为根的子树中,都满足条件,且 
 所在连通块大小为 
 的方案数。
转移类似背包,加入一个子树  时枚举断不断边。
- 断边:无论 连通块大小多大,都是合法方案。那么记 ,用 乘到 中去。 
- 不断边:枚举 的连通块大小 ,用 贡献到 ,注意 01 背包的转移顺序。 
特别注意加入子树时要按照子树大小从小到大加入保证时间复杂度。总时间复杂度 。
代码:
#include <algorithm>
#include <cstdio>
#include <cstring>
#include <vector>
const int MaxN = 2000, MaxK = 2000;
const int Mod = 998244353;
int N, K;
int Siz[MaxN + 5], Fa[MaxN + 5];
int F[MaxN + 5][MaxK + 5];
std::vector<int> Gr[MaxN + 5];
inline int add(int x, int y) { return (x += y) >= Mod ? x - Mod : x; }
inline int sub(int x, int y) { return (x -= y) < 0 ? x + Mod : x; }
inline int mul(int x, int y) { return 1LL * x * y % Mod; }
inline int pw(int x, int y) { int z = 1; for (; y; y >>= 1, x = mul(x, x)) if (y & 1) z = mul(z, x); return z; }
inline int inv(int x) { return pw(x, Mod - 2); }
inline int sep(int x, int y) { return mul(x, inv(y)); }
inline void inc(int &x, int y = 1) { (x += y) >= Mod ? x -= Mod : 0; }
inline void dec(int &x, int y = 1) { (x -= y) < 0 ? x += Mod : 0; }
void init() {
  scanf("%d %d", &N, &K);
  for (int i = 1; i < N; ++i) {
    int u, v;
    scanf("%d %d", &u, &v);
    Gr[u].push_back(v);
    Gr[v].push_back(u);
  }
}
inline bool siz_cmp(int x, int y) { return Siz[x] < Siz[y]; }
void dfs(int u) {
  Siz[u] = 1;
  for (int i = 0; i < (int) Gr[u].size(); ++i) {
    int v = Gr[u][i];
    if (v == Fa[u]) continue;
    Fa[v] = u;
    dfs(v);
    Siz[u] += Siz[v];
  }
  std::sort(Gr[u].begin(), Gr[u].end(), siz_cmp);
  int s = 1;
  F[u][1] = 1;
  for (int i = 0; i < (int) Gr[u].size(); ++i) {
    int v = Gr[u][i];
    if (v == Fa[u]) continue;
    s += Siz[v];
    int sum = 0;
    for (int j = 1; j <= Siz[v] && j <= K; ++j) inc(sum, F[v][j]);
    for (int j = std::min(s, K); j >= 0; --j) {
      F[u][j] = mul(F[u][j], sum);
      for (int k = std::max(1, Siz[v] + j - s); k <= Siz[v] && k <= j; ++k)
        inc(F[u][j], mul(F[u][j - k], F[v][k]));
    }
  }
}
void solve() {
  dfs(1);
  int ans = 0;
  for (int i = 0; i <= K; ++i) inc(ans, F[1][i]);
  printf("%d\n", ans);
}
int main() {
  init();
  solve();
  return 0;
}  查看6道真题和解析
查看6道真题和解析