Water Tree CodeForces 343D 树链剖分+线段树
题意
给定一棵n个n-1条边的树,起初所有节点权值为0。
然后m个操作, 1 x:把x为根的子树的点的权值修改为1; 2 x:把x结点到根路径上的点修改为0; 3 x:查询结点x的值。
解题思路
这个因为是在树上进行的操作,所以首先需要把树进行一些转化,比如使用dfs序列转变成一维的,这样方便使用线段树或则树状数组来进行操作。但是因为这里的操作2需要把x节点和它的父节点赋值为0,所以需要树链剖分来进行处理。
关于树链剖分的讲解可以参照我的代码,如果是初学者的话,我有一篇博文专门总结了一些优秀的关于树链剖分的文章,可以参考,点我进去
代码实现
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef long long ll;
const int maxn=5e5+7;
struct node{
int l, r;
int val, lazy; //val表示在l到r的范围内是不是都有水,lazy就是标记了
}t[maxn<<2]; //线段树的基本单元
struct edge{
int to, next;
}e[maxn<<1]; //采用链式向前星的形式来存边
int son[maxn], size[maxn], f[maxn], dep[maxn];
int in[maxn], top[maxn], cnt;
int head[maxn], len;
int n, m;
void init()
{
len=cnt=0;
for(int i=1; i<=n; i++)
head[i]=-1;
}
void add(int u, int v)
{
e[len].to=v;
e[len].next=head[u];
head[u]=len++;
}
void dfs1(int u, int fa, int depth) //这个dfs主要解决每个点的基本信息,如深度,父节点是谁,点的规模大小和它的重儿子
{
f[u]=fa;
dep[u]=depth;
size[u]=1; //点的规模包括自己和自己的所有子儿子。
for(int i=head[u]; i!=-1; i=e[i].next)
{
int v=e[i].to;
if(v==fa) //因为是无向树,所以要注意不能回去
continue;
dfs1(v, u, depth+1);
size[u]+=size[v];
if(size[v] > size[son[u]]) //一个点只有一个重儿子,就是规模最大的子儿子
son[u]=v; //记录u这个点的重儿子
}
}
void dfs2(int u, int t) //这里是dfs序,但是有所不同,在处理一个节点的子儿子时,先处理它的重儿子,也就是先处理重链。
{
top[u]=t; //链的顶部
in[u]=++cnt;//这里存储树按照新的排列方式的顺序,从1到n
if(!son[u]) //如果没有重儿子说明到了叶子节点
return ;
dfs2(son[u], t);//先处理重儿子
for(int i=head[u]; i!=-1; i=e[i].next) //下面是处理u的轻儿子
{
int v=e[i].to;
if(v==son[u] || v==f[u]) //如果遇到重儿子和它的父节点就不能再处理一遍了。
continue;
dfs2(v, v); //轻儿子的顶点就是本身
}
}
//-----------------------------这里是线段树----------------------------------
void up(int rt)
{
t[rt].val=(t[rt<<1].val && t[rt<<1|1].val);
}
void build(int rt, int l, int r)
{
t[rt].l=l;
t[rt].r=r;
t[rt].lazy=-1;
t[rt].val=0;
if(l==r)
return ;
int mid=(l+r)>>1;
build(rt<<1, l, mid);
build(rt<<1|1, mid+1, r);
}
void down(int rt)
{
if(t[rt].lazy==-1) return ;
int l=rt<<1, r=rt<<1|1;
t[l].val=t[l].lazy=t[rt].lazy;
t[r].val=t[r].lazy=t[rt].lazy;
t[rt].lazy=-1;
}
void update(int rt, int l, int r, int v)
{
if(l<=t[rt].l && t[rt].r<=r)
{
t[rt].lazy=t[rt].val=v;
return ;
}
down(rt);
int mid=(t[rt].l+t[rt].r)>>1;
if(l<=mid)
update(rt<<1, l, r, v);
if(r>mid)
update(rt<<1|1, l, r, v);
up(rt);
}
int query(int rt, int x)
{
if(t[rt].l==t[rt].r)
return t[rt].val;
down(rt);
int mid=(t[rt].l+t[rt].r)>>1;
if(x<=mid) return query(rt<<1, x);
else return query(rt<<1|1, x);
}
//--------------------线段树到此结束------------------------------------
void solve(int x) //这里是关键的一步,用来处理一个点和它的祖先们。
{
int f1=top[x]; //在这里,上面使用的top来记录一个重链的上端点的用处就体现出来了。实际上就是用来加速的
while(f1!=1)
{
update(1, in[f1], in[x], 0);
x=f[x];
f1=top[x];
}
update(1, 1, in[x], 0); //不要忘记这一步。
}
int main()
{
scanf("%d", &n);
init();
int x, y;
for(int i=1; i<n; i++)
{
scanf("%d%d", &x, &y);
add(x, y);
add(y, x);
}
dfs1(1, 0, 1);
dfs2(1, 1);
build(1, 1, n);
scanf("%d", &m);
int op;
for(int i=1; i<=m; i++)
{
scanf("%d%d", &op, &x);
if(op==1)
update(1, in[x], in[x]+size[x]-1, 1);
else if(op==2)
solve(x);
else printf("%d\n", query(1, in[x]));
}
return 0;
}