[CC-ADJLEAF2]Adjacent Leaves
[CC-ADJLEAF2]Adjacent Leaves
题目大意:
给定一棵有根树,考虑从根开始进行DFS,将所有叶子按照被遍历到的顺序排列得到一个序列。
定义一个叶子集合合法,当且仅当存在一种DFS的方式使得这个叶子集合在序列中的出现位置是一个连续子串。
给出一个\(n(n\le5\times10^5)\)个点的无根树,\(m(m\le5\times10^5)\)次询问,求以\(R\)为根的情况下叶子集合\(S(\sum|S|\le5\times10^5)\)是否合法。
思路:
定义\(leaf[x]\)为以\(x\)为根的子树内叶子结点的个数,\(cnt[x]\)为以\(x\)为根的子树内属于\(S\)的点的个数。
定义以\(x\)为根的子树是“不满的”,当且仅当\(cnt[x]\ne0\)且\(cnt[x]\ne leaf[x]\)。
对于一个结点\(x\),若其有\(\ge3\)个子树是不满的,或其有\(2\)个子树不满,且\(cnt[x]\ne|S|\),则\(S\)不合法。
对于每次询问,用一次\(\mathcal O(n)\)的树形DP求解,我们就得到了一个\(\mathcal O(nm)\)的做法。
由于\(\sum|S|\le5\times10^5\),我们可以建立虚树,将每次DFS的点数缩小为\(|S|\),此时一次DP的时间复杂度为\(\mathcal O(|S|\log n)\),总时间复杂度\(\mathcal O(\sum|S|\log n)\)。
源代码:
#include<cstdio>
#include<cctype>
#include<vector>
#include<algorithm>
inline int getint() {
register char ch;
while(!isdigit(ch=getchar()));
register int x=ch^'0';
while(isdigit(ch=getchar())) x=(((x<<2)+x)<<1)+(ch^'0');
return x;
}
const int N=5e5+1,logN=19;
std::vector<int> e[N],node;
inline void add_edge(const int &u,const int &v) {
e[u].push_back(v);
e[v].push_back(u);
}
int r,dep[N],anc[N][logN],dfn[N],s[N],leaf[N],tot;
inline int lg2(const float &x) {
return ((unsigned&)x>>23&255)-127;
}
void dfs(const int &x,const int &par) {
anc[x][0]=par;
dfn[x]=++tot;
dep[x]=dep[par]+1;
leaf[x]=e[x].size()==1;
for(register int i=1;i<=lg2(dep[x]);i++) {
anc[x][i]=anc[anc[x][i-1]][i-1];
}
for(auto &y:e[x]) {
if(y==par) continue;
dfs(y,x);
leaf[x]+=leaf[y];
}
}
inline int lca(int x,int y) {
if(dep[x]<dep[y]) std::swap(x,y);
for(register int i=lg2(dep[x]-dep[y]);i>=0;i--) {
if(dep[anc[x][i]]>=dep[y]) {
x=anc[x][i];
}
}
for(register int i=lg2(dep[x]);i>=0;i--) {
if(anc[x][i]!=anc[y][i]) {
x=anc[x][i];
y=anc[y][i];
}
}
return x==y?x:anc[x][0];
}
inline bool is_anc(const int &x,int y) {
//x shi y de zuxian
if(dep[x]>dep[y]) return false;
for(register int i=lg2(dep[y]-dep[x]);i>=0;i--) {
if(dep[anc[y][i]]>=dep[x]) {
y=anc[y][i];
}
}
return x==y;
}
inline int jump(int x,const int &d) {
//zhaodao x shendu wei d de zuxian
for(register int i=lg2(dep[x]-d);i>=0;i--) {
if(dep[anc[x][i]]>=d) {
x=anc[x][i];
}
}
return x;
}
inline int near_ch(const int &x,const int &y) {
//zhaodao x zuijin de zijiedian, bingqie shi y de zuxian
if(is_anc(x,y)) {
return jump(y,dep[x]+1);
} else {
return anc[x][0];
}
}
inline int count_leaves(const int &x) {
if(x==r) return leaf[1];
if(is_anc(x,r)) {
return leaf[1]-leaf[jump(r,dep[x]+1)];
} else {
return leaf[x];
}
}
class AuxTree {
//auxiliary tree
private:
int stk[N],top,cnt[N];
std::vector<int> e[N];
void add_edge(const int &u,const int &v) {
e[u].push_back(v);
e[v].push_back(u);
}
public:
void build(int s[],const int &r) {
const int n=s[0]+1;
s[n]=r;
std::sort(&s[1],&s[n]+1,
[](const int &i,const int &j) {
return dfn[i]<dfn[j];
}
);
stk[top=1]=s[1];
node.push_back(s[1]);
for(register int i=2;i<=n;i++) {
const int &x=s[i],p=lca(x,stk[top]);
while(dfn[p]<dfn[stk[top]]) {
if(dfn[p]>=dfn[stk[top-1]]) {
add_edge(p,stk[top--]);
if(stk[top]!=p) {
stk[++top]=p;
node.push_back(p);
}
break;
} else {
add_edge(stk[top],stk[top-1]);
top--;
}
}
stk[++top]=x;
node.push_back(x);
}
for(;top>1;top--) {
add_edge(stk[top],stk[top-1]);
}
}
void dfs(const int &x,const int &par) {
int tmp=0;
cnt[x]=x!=r&&e[x].size()==1;
for(auto &y:e[x]) {
if(y==par) continue;
dfs(y,x);
cnt[x]+=cnt[y];
tmp+=cnt[y]<count_leaves(near_ch(x,y));
}
if(tmp>2||(tmp==2&&cnt[x]!=s[0])) throw 0;
}
void clear() {
for(auto &x:node) {
e[x].clear();
}
node.clear();
}
};
AuxTree t;
int main() {
const int n=getint(),m=getint();
for(register int i=1;i<n;i++) {
add_edge(getint(),getint());
}
dfs(1,0);
for(register int i=0;i<m;i++) {
r=getint();
for(register int i=0;i<=s[0];i++) {
s[i]=getint();
}
t.build(s,r);
try {
t.dfs(r,0);
puts("YES");
} catch(...) {
puts("NO");
}
t.clear();
}
return 0;
}
[CC-ADJLEAF2]Adjacent Leaves的更多相关文章
- TOJ 4008 The Leaf Eaters(容斥定理)
Description As we all know caterpillars love to eat leaves. Usually, a caterpillar sits on leaf, eat ...
- TOJ 4008 The Leaf Eaters
|A∪B∪C|=|A|+|B|+|C|-|A∩B|-|A∩C|-|B∩C|+|A∩B∩C| 这个是集合的容斥,交集差集什么的,这个在概率论经常用到吧 4008: The Leaf Eaters T ...
- Tarjan算法求解桥和边双连通分量(附POJ 3352 Road Construction解题报告)
http://blog.csdn.net/geniusluzh/article/details/6619575 在说Tarjan算法解决桥和边双连通分量问题之前我们先来回顾一下Tarjan算法是如何 ...
- 03-树2 List Leaves
二叉树及其遍历 一遍AC,挺开心的hhh~ 简单讲下思路:叶子,顾名思义就是没有左右子树的结点.由于题目要求,叶子结点的输出顺序是从上往下,从左往右.所以用层序遍历法. 当然,这里先找到root树的根 ...
- 03-树1. List Leaves (25)
Given a tree, you are supposed to list all the leaves in the order of top down, and left to right. I ...
- List Leaves
Given a tree, you are supposed to list all the leaves in the order of top down, and left to right. I ...
- 366. Find Leaves of Binary Tree输出层数相同的叶子节点
[抄题]: Given a binary tree, collect a tree's nodes as if you were doing this: Collect and remove all ...
- CF&&CC百套计划4 Codeforces Round #276 (Div. 1) E. Sign on Fence
http://codeforces.com/contest/484/problem/E 题意: 给出n个数,查询最大的在区间[l,r]内,长为w的子区间的最小值 第i棵线段树表示>=i的数 维护 ...
- 7-8 List Leaves(25 分)
Given a tree, you are supposed to list all the leaves in the order of top down, and left to right. I ...
随机推荐
- C++设计模式——观察者模式(转)
前言 之前做了一个性能测试的项目,就是需要对现在的产品进行性能测试,获得测试数据,然后书写测试报告,并提出合理化的改善意见.项目很简单,我们获得了一系列性能测试数据,对于数据,我们需要在Excel中制 ...
- 如何使用Scrapy框架实现网络爬虫
现在用下面这个案例来演示如果爬取安居客上面深圳的租房信息,我们采取这样策略,首先爬取所有租房信息的链接地址,然后再根据爬取的地址获取我们所需要的页面信息.访问次数多了,会被重定向到输入验证码页面,这个 ...
- 该问题是需要导包!!!需要pom中添加依赖The absolute uri: http://java.sun.com/jsp/jstl/core cannot be resolved in either web.xml or the jar files deployed with this application
<!-- https://mvnrepository.com/artifact/org.apache.taglibs/taglibs-standard-impl --><depend ...
- Atom插件下载失败解决办法
转自:http://www.cnblogs.com/20145221GQ/p/5334762.html#正题 一般方法(Atom自动安装) 打开Atom >> Packages >& ...
- sqoop无法导出parquet文件到mysql
1.问题描述 在CDH集群中我们需要将Hive表的数据导入到RDBMS数据库中,使用Sqoop工具可以方便的将Hive表数据抽取到RDBMS数据库中,在使用Sqoop抽取Hive Parquet表时作 ...
- jmeter4.x centos7部署笔记
1. jmeter依赖 java8或以上版本 安装 java : 参考 https://tecadmin.net/install-java-8-on-centos-rhel-and-fedora/ ...
- 用webstorm搭建vue项目
本文只针对新手. 首先要明白几个名词(概念). Node.js: Node.js 是一个基于 Chrome V8 引擎的 JavaScript 运行环境. Node.js 使用了一个事件驱动.非阻塞式 ...
- Tomcat模型结构
一.请求过程 Tomca的两大组件:Connecter和Container Connecter组件 1.Connecter将在某个指定的端口上侦听客户请求,接收浏览器的发过来的 tcp 连接请求,创建 ...
- [转]解决-Dmaven.multiModuleProjectDirectory system property is not set. Check $M2_HOME environment variable and mvn script match.
来源:http://www.cnblogs.com/sprinng/p/5141233.html 1.添加M2_HOME的环境变量 2.Preference->Java->Installe ...
- eclipse4.2版本下面安装ADT,安装已经完成了,但没有ADT的那个图标显示
如果安装过程没错,直接在Eclipse ->window->customize Perpective->Command Groups Availability 勾选andorid 选 ...