NC20811 蓝魔法师
题目
题目描述
“你,你认错人了。我真的,真的不是食人魔。”--蓝魔法师
给出一棵树,求有多少种删边方案,使得删后的图每个连通块大小小于等于k,两种方案不同当且仅当存在一条边在一个方案中被删除,而在另一个方案中未被删除,答案对998244353取模
输入描述
第一行两个整数n,k, 表示点数和限制
2 <= n <= 2000, 1 <= k <= 2000
接下来n-1行,每行包括两个整数u,v,表示u,v两点之间有一条无向边
保证初始图联通且合法
输出描述
共一行,一个整数表示方案数对998244353取模的结果
示例1
输入
5 2
1 2
1 3
2 4
2 5
输出
7
题解
知识点:树形dp,背包dp,计数dp。
显然是个树上背包,但涉及组合计数。
设 \(dp[u][i]\) 表示以 \(u\) 为根的子树划分成大小不大于 \(k\) 连通块,且 \(u\) 所在连通块大小为 \(i\) 的方案数。这道题要剪枝,转移方程我用了刷表法。下面代码有打表和刷表的对比,明显刷表法更容易写出最佳的循环边界。转移方程为:
\begin{array}{l}
dp[u][i+j] &= dp[u][i + j] + dp[u][i] \cdot dp[v][j]\\
dp[u][i] &= dp[u][i] + \sum dp[v][j]
\end{array}
\right .
\]
- 连接 \((u,v)\) , \(dp[u][i]\) 与 \(dp[v][j]\) 用乘法原理合并,最后得到 \(dp[u][i+j]\) ,显然 \(i+j\) 要小于等于 \(k\) 。 $i \in [1,\min(sz[u],k)] $ , \(j \in [1,\min (sz[v],k)]\) ,两者都要倒序遍历保证 \(i+j\) 是倒序的。
- 断开 \((u,v)\) , \(dp[u][i]\) 的方案加上 \(dp[v][j]\) 的所有方案数。为了防止 \(dp[u][i]\) 的更新影响 \(i+j\) ,因此这个更新要在一次 \(i\) 的更新最后执行。
时间复杂度 \(O(nk^2)\)
空间复杂度 \(O(nk)\)
代码
#include <bits/stdc++.h>
using namespace std;
const int mod = 998244353;
int n, k;
vector<int> g[2007];
int dp[2007][2007], sz[2007];
/* void dfs(int u, int fa) {
dp[u][1] = 1;
sz[u] = 1;
for (auto v : g[u]) {
if (v == fa) continue;
dfs(v, u);
sz[u] += sz[v];
int sumv = 0;
for (int i = 1;i <= sz[v];i++) sumv = (sumv + dp[v][i]) % mod;
for (int i = min(sz[u], k);i >= 1;i--) {
dp[u][i] = 1LL * dp[u][i] * sumv % mod;///断开
for (int j = max(1, sz[v] + i - sz[u]);j <= min({ i, k,sz[v] });j++) ///链接
dp[u][i] = (dp[u][i] + 1LL * dp[u][i - j] * dp[v][j]) % mod;
}
}
} */
///复杂度和常数比上面一种方式少一点,甚至比上面边界好写很多
void dfs(int u, int fa) {
dp[u][1] = 1;
sz[u] = 1;
for (auto v : g[u]) {
if (v == fa) continue;
dfs(v, u);
int sumv = 0;
for (int i = 1;i <= sz[v];i++) sumv = (sumv + dp[v][i]) % mod;
for (int i = min(sz[u], k);i >= 1;i--) {
for (int j = min(sz[v], k);j >= 1;j--) {
if (i + j > k) continue;///也可以用tmp暂存的方法,然后顺序更新,就可以直接break,否则只能倒叙
dp[u][i + j] = (dp[u][i + j] + 1LL * dp[u][i] * dp[v][j]) % mod;
}
dp[u][i] = 1LL * dp[u][i] * sumv % mod;
}
sz[u] += sz[v];
}
}///复杂度 O(nk^2),剪枝以后复杂度不变但常数会少巨多,因为少遍历了很多不可能的状态
int main() {
std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
cin >> n >> k;
for (int i = 1;i < n;i++) {
int u, v;
cin >> u >> v;
g[u].push_back(v);
g[v].push_back(u);
}
dfs(1, 0);
int ans = 0;
for (int i = 1;i <= k;i++) ans = (ans + dp[1][i]) % mod;
cout << ans << '\n';
return 0;
}
NC20811 蓝魔法师的更多相关文章
- 牛客网 Wannafly挑战赛27 蓝魔法师
蓝魔法师 链接: https://www.nowcoder.com/acm/contest/215/C 来源:牛客网 题目描述 "你,你认错人了.我真的,真的不是食人魔."--蓝魔 ...
- Wannafly挑战赛27 C蓝魔法师
链接Wannafly挑战赛27 C蓝魔法师 给出一棵树,求有多少种删边方案,使得删后的图每个连通块大小小于等于\(k\),\(n,k\leq 2*10^3\) 假设我们正在考虑\(i\)这个子树,那么 ...
- Wannafly挑战赛27
Wannafly挑战赛27 我打的第一场$Wannafly$是第25场,$T2$竟然出了一个几何题?而且还把我好不容易升上绿的$Rating$又降回了蓝名...之后再不敢打$Wannafly$了. 由 ...
- BZOJ-5055-膜法师(离散化+树状数组)
Description 在经历过1e9次大型战争后的宇宙中现在还剩下n个完美维度, 现在来自多元宇宙的膜法师,想偷取其中的三个维度为伟大的长者续秒, 显然,他能为长者所续的时间,为这三个维度上能量的乘 ...
- [BZOJ 5055]膜法师
Description 在经历过1e9次大型战争后的宇宙中现在还剩下n个完美维度, 现在来自多元宇宙的膜法师,想偷取其中的三个维度为伟大的长者续秒, 显然,他能为长者所续的时间,为这三个维度上能量的乘 ...
- bzoj5055 膜法师
Description 在经历过1e9次大型战争后的宇宙中现在还剩下n个完美维度, 现在来自多元宇宙的膜法师,想偷取其中的三个维度为伟大的长者续秒, 显然,他能为长者所续的时间,为这三个维度上能量的乘 ...
- BZOJ_5055_膜法师_树状数组+离散化
BZOJ_5055_膜法师_树状数组+离散化 Description 在经历过1e9次大型战争后的宇宙中现在还剩下n个完美维度, 现在来自多元宇宙的膜法师,想偷取其中的三个维度为伟大的长者续秒, 显然 ...
- bzoj 5055: 膜法师——树状数组
Description 在经历过1e9次大型战争后的宇宙中现在还剩下n个完美维度, 现在来自多元宇宙的膜法师,想偷取其中的三个维度为伟大的长者续秒, 显然,他能为长者所续的时间,为这三个维度上能量的乘 ...
- bzoj 5055: 膜法师 -- 树状数组
5055: 膜法师 Time Limit: 10 Sec Memory Limit: 128 MB Description 在经历过1e9次大型战争后的宇宙中现在还剩下n个完美维度, 现在来自多元宇 ...
- JZOJ.5280【NOIP2017模拟8.15】膜法师
Description
随机推荐
- c#(nanoframework)安装单片机环境;如何使用c#写类似于c的单片机驱动
.NET nanoFramework 安装教程 准备材料 esp32单片机(支持wifi蓝牙) 安卓数据线(需要支持传输) 注意!请先安装esp32驱动程序 ESP32驱动链接 安装 .NET n ...
- 【Hash】字符串哈希
Hash 的核心思想在于,将输入映射到一个值域较小.可以方便比较的范围,典型的用法就是将资源紧张的设备中的不定长字符串转化为定长整数,以达到节省空间的目的 如:printf("This is ...
- WebApi允许跨域
services.AddCors(options => { options.AddPolicy("abc", builder => { //App:CorsOrigin ...
- [转帖]Shell字符串拼接(连接、合并)
http://c.biancheng.net/view/1114.html 在脚本语言中,字符串的拼接(也称字符串连接或者字符串合并)往往都非常简单,例如: 在 PHP 中,使用.即可连接两个字符串: ...
- [转帖]Linux make: g++: Command not found
https://www.cnblogs.com/kerrycode/p/4748606.html Linux使用make命令时遇到"make: g++: Command not found& ...
- systemctl 关闭图形界面的办法
开机以命令模式启动,执行: systemctl set-default multi-user.target 开机以图形界面启动,执行: systemctl set-default graphica ...
- vue 半场动画进入状态
<style> .box{ width: 30px; height: 30px; border-radius: 50%; background: red; } </style> ...
- 如何用etcdctl产生分布式环境中的递增ID
作者:张富春(ahfuzhang),转载时请注明作者和引用链接,谢谢! cnblogs博客 zhihu Github 公众号:一本正经的瞎扯 用golang代码当然很简单,我这里是在shell环境中, ...
- TextBrewer:融合并改进了NLP和CV中的多种知识蒸馏技术、提供便捷快速的知识蒸馏框架、提升模型的推理速度,减少内存占用
TextBrewer:融合并改进了NLP和CV中的多种知识蒸馏技术.提供便捷快速的知识蒸馏框架.提升模型的推理速度,减少内存占用 TextBrewer是一个基于PyTorch的.为实现NLP中的知识蒸 ...
- 【深度学习项目一】全连接神经网络实现mnist数字识别
相关文章: [深度学习项目一]全连接神经网络实现mnist数字识别 [深度学习项目二]卷积神经网络LeNet实现minst数字识别 [深度学习项目三]ResNet50多分类任务[十二生肖分类] 『深度 ...