CF1328E Tree Queries
CF1328E Tree Queries
应该还是比较妙的
题意
给你一个树,然后多次询问
每次询问给出一堆节点,问你是否能找到一个从根出发的链,是的对于给出的每个节点,都能找出链上的点,是的他们的距离小于等于\(1\)
\(n\leq 2\cdot 10^5,m\leq 2\cdot 10^5,\sum k\leq 2\cdot 10^5\)
其中\(m\)是询问次数,\(k\)是每次给出的点数
首先,一个点要想符合题目的条件,无非有两种情况
一种是就在链上,这个好说
另一种是距离链上的点距离为\(1\),那么,肯定是他的父亲在链上
所以,我们可以把每个给出的点转换成他的父亲,问题就变成看:能否找到一个链,是的这些点都在链上
这样的转换对第一种肯定是没有影响的,一个点在链上,他的父亲肯定也在链上
所以考虑转换后的问题
我首先想到的是,找出深度最大的节点,对于其它每个点和它求 LCA ,如果这个 LCA 不是那个深度较小的点,说明肯定连不成一个链
这一点判断基于那个链必须从根出发
然后写完代码交上去:

#100 WA!
emmm,估计是写炸了,第一个MLE是邻接表忘开2倍了
然而并没看出错,所以考虑一种实现更简单的方法
上面求 LCA 其实可以理解为从下向上考虑,那么还有一种从上往下考虑的方法:
先把每个点按深度从小到大排序,然后如果p[i+1]不在p[i]的子树里,那么肯定连不成一个链
至于怎么判断也很简单,记录一个dfs序和每个节点子树大小就行,具体见代码的check函数
\(\texttt{code.}\)
#include<cstdio>
#include<algorithm>
#include<iostream>
#include<cmath>
#include<iomanip>
#include<cstring>
#define reg register
#define EN std::puts("")
#define LL long long
inline int read(){
int x=0,y=1;
char c=std::getchar();
while(c<'0'||c>'9'){if(c=='-') y=0;c=std::getchar();}
while(c>='0'&&c<='9'){x=x*10+(c^48);c=std::getchar();}
return y?x:-x;
}
int n,m;
int fir[200006],nex[400006],to[400006],tot;
int p[200006],fa[200006],size[200006];
int dfn[200006],deep[200006],dfscnt;
inline void add(int a,int b){
to[++tot]=b;
nex[tot]=fir[a];fir[a]=tot;
}
void dfs(int u,int fat){
fa[u]=fat;deep[u]=deep[fat]+1;dfn[u]=++dfscnt;size[u]=1;
for(reg int i=fir[u];i;i=nex[i])if(to[i]!=fat) dfs(to[i],u),size[u]+=size[to[i]];
}
inline int check(int a,int b){//看b是不是在a的子树
if(dfn[b]<dfn[a]) return 0;
if(dfn[b]>=dfn[a]+size[a]) return 0;
//这里一定是>=,如果dfn[b]=dfn[a]+size[a],也说明b并不在a的子树里,因为统计size[a]的时候a这个点也被算进去了
return 1;
}
inline int cmp(int x,int y){return deep[x]<deep[y];}
int main(){
n=read();m=read();
for(reg int a,b,i=1;i<n;i++){
a=read();b=read();
add(a,b);add(b,a);
}
reg int k;
dfs(1,1);
while(m--){
k=read();
for(reg int i=1;i<=k;i++) p[i]=fa[read()];
std::sort(p+1,p+1+k,cmp);
for(reg int i=2;i<=k;i++)
if(!check(p[i-1],p[i])) goto NO;
std::puts("YES");continue;
NO:;std::puts("NO");
}
return 0;
}
那个没A的LCA做法也放在这,也许哪天很闲的时候会再调调
记录
#include<cstdio>
#include<algorithm>
#include<iostream>
#include<cmath>
#include<iomanip>
#include<cstring>
#define reg register
#define EN std::puts("")
#define LL long long
inline int read(){
int x=0,y=1;
char c=std::getchar();
while(c<'0'||c>'9'){if(c=='-') y=0;c=std::getchar();}
while(c>='0'&&c<='9'){x=x*10+(c^48);c=std::getchar();}
return y?x:-x;
}
int n,m;
int fir[200006],nex[400006],to[400006],tot;
int p[200006];
int fa[23][200006],deep[200006];
inline void add(int a,int b){
to[++tot]=b;
nex[tot]=fir[a];fir[a]=tot;
}
void dfs(int u,int fat){
fa[0][u]=fat;deep[u]=deep[fat]+1;
for(reg int i=fir[u];i;i=nex[i])if(to[i]!=fat) dfs(to[i],u);
}
inline int check(int a,int b){
for(reg int i=20;~i;i--)
if(deep[fa[i][b]]>=deep[a]) b=fa[i][b];
return a==b;
}
int main(){
n=read();m=read();
for(reg int a,b,i=1;i<n;i++){
a=read();b=read();
add(a,b);add(b,a);
}
reg int k,maxdeep,maxdeepid;
dfs(1,1);
for(reg int i=1;i<=n;i++)
for(reg int j=1;j<=20;j++) fa[j][i]=fa[j-1][fa[j-1][i]];
while(m--){
k=read();maxdeep=0;
for(reg int i=1;i<=k;i++){
p[i]=fa[0][read()];
if(deep[p[i]]>maxdeep) maxdeep=deep[p[i]],maxdeepid=p[i];
}
for(reg int i=1;i<=k;i++)if(p[i]!=maxdeepid){
if(!check(p[i],maxdeepid)) goto NO;
}
std::puts("YES");continue;
NO:;std::puts("NO");
}
return 0;
}
CF1328E Tree Queries的更多相关文章
- E. Tree Queries 解析(思維、LCA)
Codeforce 1328 E. Tree Queries 解析(思維.LCA) 今天我們來看看CF1328E 題目連結 題目 給你一棵樹,並且給你\(m\le2e5\)個詢問(包含\(k\)個點) ...
- Codeforces 960D - Full Binary Tree Queries
960D - Full Binary Tree Queries 思路: 用move1[i]记录第i层第1种操作移动的个数(对这一层的个数取模) 用move2[i]记录第i层第2种操作移动的个数(对这一 ...
- AC日记——825G - Tree Queries
825G - Tree Queries 思路: 神题,路径拆成半链: 代码: #include <cstdio> #include <cstring> #include < ...
- CF-1328 E. Tree Queries
E. Tree Queries 题目链接 题意 给定一个树,每次询问一组点,问是否存在一条从根到某点的路径,使得该组点到该路径的最短距离不超过1 分析 从根到达某点的路径,如果覆盖到了某个点,那么一定 ...
- CF1254D Tree Queries(树链剖分)
出题人的做法是 \(O(n\sqrt{n\log n})\),结果这场结束后就被狂喷,一群人给出了 \(O(n\sqrt{n})\) 做法,甚至 \(O(n\log n)\) 都出来了-- 首先发现, ...
- Codeforces 960D Full Binary Tree Queries ( 思维 && 模拟 )
题意 : 给出一颗无限层的满二叉树,然后每个值为 X (根的 X 等于 1 ) 左孩子的值是 2*X,右孩子的值是 2*X+1 ,现在有两种操作,(1, x,k) 表示将 x 所在层的所有节点整体向右 ...
- CodeForces 825G"Tree Queries"(选根建树)
传送门 •参考资料 [1]:CodeForces 825G Educational Round #25 G :建树选根大法+O1大法+iostream解绑了还是慢 •题意 给定一颗包含 n 个节点的树 ...
- CF825G Tree Queries
[题意] 一棵树有 n个节点,初始均为白色,有两种操作: 1. 1 x 代表把结点 x 设置为黑色 2. 2 x 代表查询 x 到树上任意一个黑色结点的简单路径上的编号最小的结点的编号 输入 t 和 ...
- Codeforces 1606F - Tree Queries(虚树+树形 dp)
Codeforces 题面传送门 & 洛谷题面传送门 显然我们选择删除的点连同 \(u\) 会形成一个连通块,否则我们如果选择不删除不与 \(u\) 在同一连通块中的点,答案一定更优. 注意到 ...
随机推荐
- Java第十八天,可变参数
可变参数 1.使用前提 当一个方法的参数需要多个参数,并且这些参数的类型一致时,可以使用可变参数. 2.使用方法 定义方法时使用 3.定义格式 修饰符 返回值类型 方法名(参数类型...变量名){ } ...
- C语言 文件操作(四)
1.fprintf int fprintf(FILE *stream, const char *format, ...) stream -- 这是指向 FILE 对象的指针,该 FILE 对象标识了流 ...
- String 对象-->replace() 方法
1.定义和用法 replace() 方法用于字符串替换 语法: string.replace(searchvalue,newvalue) 参数: searchvalue:被替换的字符串 newvalu ...
- 七、环回接口ip地址(逻辑接口)
loopback接口,在网络设备(一般是路由器)上是一种特殊的接口,它不是物理接口,而是一种看不见摸不着的逻辑接口(也称虚拟接口),但是对于网络设备来说却是至关重要的. 在网络设备上可以通过配置命令来 ...
- OS X10.10.3正式版和Xcode 6.3正式版下载
4.09日,OS X10.10.3 正式版 IOS8.3 正式版和 Xcode 6.3 正式版在今天发布,这是 2015 年里面,IOS 系统和 Mac OS 系统,以及 IOS 和 Mac OS 专 ...
- java 第七周课后作业0417
定义一个矩形类Rectangle:(知识点:对象的创建和使用)1 定义三个方法:getArea()求面积.getPer()求周长,showAll()分别在控制台输出长.宽.面积.周长.2 有2个属性: ...
- 使用SVG内置API计算图形或点经过transform之后的新坐标
一个应用场景是,点击一条路径,显示该路径的控制点.因为有transform变形( 平移.缩放.倾斜.旋转等变换),所以获取变形后的新坐标需要计算. 纯数学的方法,就是用2D变换矩阵的一些公式去运算,过 ...
- IE各版本CSS Hack(兼容性处理)语法速查表
为了兼容IE各个版本,需要在CSS中添加额外的代码,比如以前常用的_width.之所以工作,是因为浏览器会忽略不能解析的样式规则,因此举个例子来说,把_width写在width下面,对于非IE浏览器会 ...
- day28作业
import os import uuid import pickle from conf import settings class School: def __init__(self,name,a ...
- 14. 最长公共前缀----LeetCode
编写一个函数来查找字符串数组中的最长公共前缀. 如果不存在公共前缀,返回空字符串 "". 示例 1: 输入: ["flower","flow" ...