小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] ...
随机推荐
- tensorflow版本的bert模型 GPU的占用率为100%而其利用率为0%
Notice: 本方法只是解决问题的一种可能,不一定百分百适用,出现这个问题还有很多其他原因,这个可以作为解决的一种尝试!!! 经过检查发现,是由于激活环境的原因 使用 conda activate ...
- BUUCTF-乌镇峰会种图
乌镇峰会种图 16进制拖到底一看便知
- 线程池:ThreadPoolExcutor源码阅读
ThreadPoolExcutor源码流程图:(图片较大,下载再看比较方便) 线程池里的二进制奥秘 前言: 线程池的五种状态state(RUNNING.SHUTDOWN.STOP.TIDYING.TE ...
- Vue2自定义插件的写法-Vue.use()
最近在用vue2完善一个项目,顺便温习下vue2的基础知识点! 有些知识点恰好没用到时间一长就会淡忘,这样对自己是一种损失. 定义一个对象 对象里可以有任何内容 但install的函数是必不可少的,因 ...
- RPA应用场景-营业收入核对
场景概述营业收入核对 所涉系统名称 SAP ,Excel,门店业务系统 人工操作(时间/次) 4 小时 所涉人工数量 6 操作频率每日 场景流程 1.每日13点起进入SAP查询前一日营业收入记账情况: ...
- 《HelloGitHub》第 75 期
兴趣是最好的老师,HelloGitHub 让你对编程感兴趣! 简介 HelloGitHub 分享 GitHub 上有趣.入门级的开源项目. https://github.com/521xueweiha ...
- NC204382 中序序列
NC204382 中序序列 题目 题目描述 给定一棵有 \(n\) 个结点的二叉树的先序遍历与后序遍历序列,求其中序遍历序列. 若某节点只有一个子结点,则此处将其看作左儿子结点 示例1 输入 5,[3 ...
- 跨平台(32bit和64bit)的 printf 格式符 %lld 输出64位的解决方式
问题描述 在 C/C++ 开发中,使用 printf 打印 64 位变量比较常用,通常在 32 位系统中使用 %lld 输出 64 位的变量,而在 64 位系统中则使用 %ld: 如果在 32 位系统 ...
- Tapdata “设擂招贤”携手 LeetCode 举办全球极客技术竞赛
2021年11月28日 Tapdata 专场全球极客技术竞赛将在 LeetCode 平台开赛,面向程序员"设擂招贤",打擂成功的前50名挑战者将优先获得 Tapdata 高端技 ...
- LEACH分簇算法实现和能量控制算法实现
一.前言 1.在给定WSN的节点数目(100)前提下,节点随机分布,按照LEACH算法,实现每一轮对WSN的分簇.记录前K轮(k=10)时,网络的分簇情况,即每个节点的角色(簇头或簇成员).标记节点之 ...