poj3321(dfs序+线段树)

  题意就是给一棵树,然后有2种操作,一是问以x为根节点的子树有多少苹果,二是更改某点的苹果数量。

 做法就是用dfs序把树形结构转化成线性结构也就是区间,然后用线段树维护。先要求出每个点的in和out值,然后就给每个点分配了一个新编号了,就是它的in[i],然后dfs序有个特点就是以x为根节点的子树里的节点的dfs序是连续的,所以in[x]~out[x]就对应着这颗子树的信息,查询的话对in[x]~out[x]区间操作。

#include<stdio.h>
#include<iostream>
#include<algorithm>
using namespace std;
#define ls rt<<1
#define rs (rt<<1)|1
#define fuck(x) cout<<#x<<"     "<<x<<endl;
const int maxn=1e5+10;
struct node
{
    int u,v,nxt;
}e[maxn<<1];
int head[maxn],in[maxn],out[maxn],order[maxn],sum[maxn<<2],tim;
void pushup(int rt)
{
    sum[rt]=sum[ls]+sum[rs];
}
void build(int rt,int L,int R)
{
    if(L==R)
    {
        sum[rt]=1;
        return ;
    }
    int mid=(L+R)>>1;
    build(ls,L,mid);
    build(rs,mid+1,R);
    pushup(rt);
}
void update(int rt,int L,int R,int pos)
{
    if(L==R)
    {
        if(sum[rt])
            sum[rt]=0;
        else
            sum[rt]=1;
        return ;
    }
    int mid=(L+R)>>1;
    if(pos<=mid)
        update(ls,L,mid,pos);
    else
        update(rs,mid+1,R,pos);
    pushup(rt);
}
int query(int rt,int L,int R,int l,int r)
{
    if(l<=L&&r>=R)
    {
        return sum[rt];
    }
    int mid=(L+R)>>1,ans=0;
    if(r<=mid)
        ans=query(ls,L,mid,l,r);
    else
        if(l>mid)
            ans=query(rs,mid+1,R,l,r);
        else
            ans=query(ls,L,mid,l,r)+query(rs,mid+1,R,l,r);
    return ans;
}
void dfs(int now,int fa)
{
    in[now]=++tim;
    for(int i=head[now];i!=-1;i=e[i].nxt)
    {
        if(e[i].v==fa) continue;
        dfs(e[i].v,now);
    }
    out[now]=tim;
}

int main()
{
    int n,q,cnt=0;
    scanf("%d",&n);
    for(int i=1;i<=n;i++) head[i]=-1;
    for(int i=1;i<n;i++){
        int u,v;
        scanf("%d%d",&u,&v);
        e[++cnt].u=u;
        e[cnt].v=v;
        e[cnt].nxt=head[u];
        head[u]=cnt;
        e[++cnt].u=v;
        e[cnt].v=u;
        e[cnt].nxt=head[v];
        head[v]=cnt;
    }
    dfs(1,0);
    for(int i=1;i<=n;i++) order[in[i]]=i;
    build(1,1,n);
    scanf("%d",&q);
    while(q--)
    {
        char ch;
        int k;
        scanf(" %c%d",&ch,&k);
        if(ch=='C')
            update(1,1,n,in[k]);
        else
            printf("%d\n",query(1,1,n,in[k],out[k]));
    }
    return 0;
}

 

posted @ 2019-07-12 09:30  eason99  阅读(137)  评论(0编辑  收藏  举报