小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] ...
随机推荐
- Lucene从入门到实战
Lucene 在了解Lucene之前,我们先了解下全文数据查询. 全文数据查询 我们的数据一般分为两种:结构化数据和非结构化数据 结构化数据:有固定格式或有限长度的数据,如数据库中的数据.元数据 非结 ...
- LVS简单搭建(一)
先说一个坑: 192.168.65.110为虚拟ip,在主节点设置(ifconfig eth0:8 192.168.65.110/24),其他子节点要与主节点保持一致(ifconfig lo:3 ...
- JavaScript 语言入门
目录 JavaScript 介绍 JavaScript 和 和 html 代码的结合方式 第一种方式 第二种方式 4.变量 关系(比较)运算 逻辑运算 数组(重点) 函数(重点) 函数的二种定义方式 ...
- FICO 常用事务码
1.SAP配置流程 1.定义,定义组织,概念,比如FI中定义公司代码,会计科目表,年度变式.SAP中有大量的定义过程. 2.分配,把会计科目表/公司/年度变式等参数分配到公司代码,逻辑组织,基本实现框 ...
- CompletableFuture的入门
runAsync 和 supplyAsync runAsync接受一个Runable的实现,无返回值 CompletableFuture.runAsync(()->System.out.prin ...
- Spring框架系列(9) - Spring AOP实现原理详解之AOP切面的实现
前文,我们分析了Spring IOC的初始化过程和Bean的生命周期等,而Spring AOP也是基于IOC的Bean加载来实现的.本文主要介绍Spring AOP原理解析的切面实现过程(将切面类的所 ...
- 关于 用fscanf读文件,把文件中用##分割的内容分开
今天呀,被学弟问了一个问题 文件里存的是"123##456##0##1644444.....##" 为什么用fscanf(fp, "%s##%s......", ...
- 比起网易有数BI,也许这款数据可视化软件更适合你!
有数BI是网易推出的面向企业客户的可视化敏捷BI产品.拥有数据填报和自助式商业智能分析产品,提供网页端和手机端应用,帮助客户快速实现数据填报.多维分析.大数据探索.实时大数据展示和成员分享. 山海鲸可 ...
- java的elasticsearch做高亮显示
import org.apache.commons.lang3.reflect.FieldUtils;import org.elasticsearch.action.search.SearchResp ...
- day02_基本语法
基本语法 学习目标: 1. 数据类型 2. 变量 3. 编码介绍 4. 标识符和关键字 5. 字符串类型 6. 数据类型转化 7. 进制转换 8. 运算符 一.数据类型 什么是数据类型? 在开发软件的 ...