小A的树 - 树形DP
题面
1
9 4
4 1
1 5
1 2
3 2
3 6
6 7
6 8
9 6
0 1 0 1 0 0 1 0 1
3 2
7 3
4 0
9 5
YES
YES
NO
NO
题解
n <= 5000 可以用DP做
把答案都算出来存在一个数组f[x][y]中,表示当询问为 x 和 y 时能不能达到,查询时就可以直接访问了。
令dp[x][y][2]记录以 x 为根的子树中选 y 个点(包括 x ),最大的黑点数max以及最小的黑点数min,求出来后把 f[y][min~max] 全都赋为 1。
接下来我们要证明两个结论:
min~max中间的值都可取
考虑绿色的部分是最小值的联通子树,蓝色部分是最大值的联通子树(重合部分为渐变色)
只需要证明当我们从最小值过渡到最大值时,它是连续的就行,从最小值过渡到最大值,由于选的点数不变,一直为 y ,所以每次在绿色部分缩回一个点,蓝色部分扩张一个点,绿色部分会让累计黑点数减 1 或 0,蓝色部分会让累计黑点数加 1 或 0 ,所以每次变化会使累计数 +1,-1或不变,在整数上是连续的。
复杂度为O(n^2)
dp的转移实际上是在做树上背包,枚举儿子的时候每次是双层循环,第一层是前几个儿子子树累计的size,第二层是新的儿子子树的size,所以也就相当于在前几个儿子的所有子树中找一个点,再在新儿子子树中找一个点,然后在 lca 上也就是 x 上转移,那么宏观上来说就是每两个点计算一次,复杂度O(n^2),f 数组处理可以差分,也是O(n^2)。
最后,为了卡空间,我把最大值和最小值存在一个int中了。
CODE
#include<map>
#include<stack>
#include<queue>
#include<vector>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
#define MAXN 5005
#define LL long long
#define DB double
#define ENDL putchar('\n')
#define lowbit(x) ((-x) & (x))
LL read() {
LL f = 1,x = 0;char s = getchar();
while(s < '0' || s > '9') {if(s=='-')f = -f;s = getchar();}
while(s >= '0' && s <= '9') {x=x*10+(s-'0');s = getchar();}
return f * x;
}
const int MOD1 = 1000000007;
const int sq = 10000;
int n,m,i,j,s,o,k;
vector<int> g[MAXN];
int f[MAXN][MAXN];
int dp[MAXN][MAXN];
int c[MAXN],son[MAXN];
void dfs(int x,int fa) {
for(int i = 0;i <= n;i ++) dp[x][i] = n+1;
dp[x][1] = c[x] * sq + c[x];
son[x] = 1;
for(int i = 0;i < (int)g[x].size();i ++) {
int y = g[x][i];
if(y != fa) {
dfs(y,x);
for(int k = son[x] + son[y];k > 1;k --) {
for(int j = max(1,k-son[x]);j <= son[y] && j < k;j ++) {
int dp0,dp1;
dp0 = min(dp[x][k]%sq,(dp[x][k-j]%sq) + (dp[y][j]%sq));
dp1 = max(dp[x][k]/sq,(dp[x][k-j]/sq) + (dp[y][j]/sq));
dp[x][k] = dp1 * sq + dp0;
}
}
son[x] += son[y];
}
}
for(int i = 1;i <= son[x];i ++) {
int ll = dp[x][i]%sq,rr = dp[x][i]/sq;
if(ll <= rr) f[i][ll] ++,f[i][rr+1] --;
}
return ;
}
int main() {
// freopen("tree.in","r",stdin);
// freopen("tree.out","w",stdout);
int T = read();
while(T --) {
n = read();m = read();
memset(f,0,sizeof(f));
memset(c,0,sizeof(c));
memset(dp,0,sizeof(dp));
memset(son,0,sizeof(son));
for(int i = 1;i <= n;i ++) g[i].clear();
for(int i = 1;i < n;i ++) {
s = read();o = read();
g[s].push_back(o);
g[o].push_back(s);
}
for(int i = 1;i <= n;i ++) {
c[i] = (bool)read();
}
dfs(1,0);
for(int i = 1;i <= n;i ++) {
for(int j = 1;j <= i;j ++) {
f[i][j] += f[i][j-1];
}
}
for(int i = 1;i <= m;i ++) {
s = read();o = read();
if(s == 0 && o == 0) printf("YES\n");
else if(s < 1 || s > n || o < 0 || o > s) printf("NO\n");
else printf(f[s][o] > 0 ? "YES\n":"NO\n");
}
ENDL;
}
return 0;
}
小A的树 - 树形DP的更多相关文章
- 牛客挑战赛30 小G砍树 树形dp
小G砍树 dfs两次, dp出每个点作为最后一个点的方案数. #include<bits/stdc++.h> #define LL long long #define fi first # ...
- 【BZOJ5072】[Lydsy十月月赛]小A的树 树形DP
[BZOJ5072][Lydsy十月月赛]小A的树 题解:考虑我们从一个联通块中替换掉一个点,导致黑点数量的变化最多为1.所以我们考虑维护对于所有的x,y的最大值和最小值是多少.如果询问的y在最大值和 ...
- bzoj 5072 小A的树 —— 树形DP
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=5072 由于对于一个子树,固定有 j 个黑点,连通块大小是一个连续的范围: 所以记 f[i][ ...
- bzoj 5072 [Lydsy1710月赛]小A的树——树形dp
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=5072 发现对于每个子树,黑点个数确定时,连通块的大小取值范围一定是一段区间:所以考虑只最小化 ...
- BZOJ5072:[Lydsy1710月赛]小A的树(树形DP)
Description BZOJ只是扔了个下载链接 Solution 设$f[x][i]$表示$x$点选中$i$个黑点的最小连通块. 设$g[x][i]$表示$x$点选中$i$个黑点的最大连通块. 转 ...
- 51nod 1353 树 | 树形DP经典题!
51nod 1353 树 | 树形DP好题! 题面 切断一棵树的任意条边,这棵树会变成一棵森林. 现要求森林中每棵树的节点个数不小于k,求有多少种切法. 数据范围:\(n \le 2000\). 题解 ...
- 【BZOJ2616】SPOJ PERIODNI 笛卡尔树+树形DP
[BZOJ2616]SPOJ PERIODNI Description Input 第1行包括两个正整数N,K,表示了棋盘的列数和放的车数. 第2行包含N个正整数,表示了棋盘每列的高度. Output ...
- 【BZOJ-3572】世界树 虚树 + 树形DP
3572: [Hnoi2014]世界树 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 1084 Solved: 611[Submit][Status ...
- 【BZOJ-2286】消耗战 虚树 + 树形DP
2286: [Sdoi2011消耗战 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 2120 Solved: 752[Submit][Status] ...
随机推荐
- 2021.06.05【NOIP提高B组】模拟 总结
T1 题意:给你一个 \(n\) 个点 \(n\) 条边的有向图, 求每个店经过 \(K\) 条边后的边权和.最小边权 \(K\le 10^{10}\) 考试时:一直想着环,结果一直不知道怎么做 正解 ...
- 互联网公司实行目标管理(OKR)五点原则和基础
下面从公司文化.组织架构.管理者.落地执行和区别绩效考核五个方面,讲述了如何在公司落地目标管理(OKR),这些是实施OKR的基础,也是原则,虽然写得比较简单,其实是我过去两年多不断观察.实践和摸索的总 ...
- Node.js amqplib 连接 Rabbitmq 学习笔记
var amqp = require('amqplib'); connect([url, [socketOptions]]) var amqp = require('amqplib/callback_ ...
- Python if-else的简单表示
常见写法 a = 1 b = 1 c = 2 if a == b: print("true") elif a == c: print("false") else ...
- 3D大场景展示功能你了解多少?见详解!
裸眼3D技术的出现打破了真实与虚拟的界限,人们不仅希望能够体验奇妙的虚拟场景,也希望足不出户在短短几分钟内就能看到遍布各地的场景,希望能实时对接关键数据. 裸眼3D技术的出现打破了真实与虚拟的界限,人 ...
- 超 Nice 的表格响应式布局小技巧
今天,遇到了一个很有意思的问题,一名群友问我,仅仅使用 CSS,能否实现这样一种响应式的布局效果: 简单解析一下效果: 在屏幕视口较为宽时,表现为一个整体 Table 的样式 而当屏幕视口宽度较小时, ...
- [LINUX] 像电影里的黑客一样用 terminal 作为日常开发
目录 1.效果预览 2.具体实现 2.1 定位鼠标位置 2.2 获取屏幕位置 2.3 计算鼠标在哪个窗口 2.4 1920x1080 平铺效果设计 2.5 1280x1024 平铺效果设计 3 代码 ...
- 『现学现忘』Git后悔药 — 30、版本回退git reset --hard命令说明
git reset --hardcommit-id命令:回退到指定版本.(hard:强硬,严格的回退) 该命令不仅移动了分支中HEAD指针的位置,还将工作区和暂存区中数据也回退到了指定的版本. (提示 ...
- Solution -「SDOI2011」拦截导弹
Sol. 题目要求一个数对序列的二维最长下降子序列,我们称其为 Q.并求出每一个元素分别在可能的 Q 中出现了多少次. 直接 Dp,时间复杂度 \(O(n^2)\) 不行.考虑 CDQ 分治 ...
- .net6与英雄联盟邂逅之——根据官方LCU API制作游戏助手
看了网上很多自己开发的英雄联盟的小助手工具,当时苦于没有api,自己也想做一个.后来发现了其实拳头本身就提供了LCU API在客户端运行的时候会暴露出来. 现在我们就来了解下工具的实现. 查询数据:h ...