题意
给出一棵树,求有多少种删边方案,使得删后的图每个连通块大小小于等于k,两种方案不同当且仅当存在一条边在一个方案中被删除,而在另一个方案中未被删除,答案对998244353取模。
分析
我们定义表示为i的子树的所有联通块大小不超过k,并且i所在联通块大小为j的方案数。
对于每条边,假设为 u —— v ,我们都有两种选择,删与不删,如果删去的话,u的联通块大小不会变,如果不删的话,u的联通块大小就会变成两个联通块大小之和。
树形dp+背包
#include <bits/stdc++.h>
using namespace std;
#define mem(a,b) memset(a,b,sizeof(a))
#define pii pair<int,int>
#define int long long
const int inf = 0x3f3f3f3f;
const int maxn = 2110;
const int M = 998244353;
int n,m,k,ok;
vector<int> v[maxn];
int dp[maxn][maxn]; //i的子树的所有联通块大小不超过k,并且i所在联通块大小为j
int sz[maxn];
void dfs(int u,int pre)
{
dp[u][1] = sz[u] = 1;
for(auto i : v[u])
{
if(i == pre) continue;
dfs(i,u);
int sum = 0;
for(int j = 1; j <= sz[i] && j <= k; j++) sum = (sum+dp[i][j])%M;
for(int j = min(sz[u],k); j >= 1; j--) //枚举u的联通块大小
{
for(int w = min(sz[i],k); w >= 1; w--) //大的先更新
{
if(j+w <= k) dp[u][j+w] = (dp[u][j+w] + dp[u][j]*dp[i][w]%M)%M;
}
//断开这条边
dp[u][j] = dp[u][j]*sum%M;
}
sz[u] += sz[i];
}
}
signed main()
{
ios::sync_with_stdio(false);cin.tie(0);
cin>>n>>k;
for(int i = 2,x,y; i <= n; i++)
{
cin>>x>>y;
v[x].push_back(y);
v[y].push_back(x);
}
dfs(1,0);
int ans = 0;
for(int i = 1; i <= k; i++) ans = (ans + dp[1][i])%M;
cout<<ans<<endl;
return 0;
} 
京公网安备 11010502036488号