P4116 Qtree3

P4116 Qtree3

#LJ-高级数据结构-01

讲真的,这题挺不错

首先发现QTREE系列,大力树剖

其次是第一个黑点,转化成pos到1路径上的最后一个黑点,对于每个重链用set维护

插入:

直接搞

查找:

跳链,对于一条链,如果第一个黑点的深度都大于u,那么这条链一定挂;否则更新答案

代码:

#include<bits/stdc++.h>
using namespace std;
const int N=100005;
int n,q;
int hed[N<<1],tal[N<<1],val[N<<1],nxt[N<<1],cnt=0;
int son[N]={0};
int size[N]={0};
int indexy=0;
int dfn[N];
int tp[N];
int id[N];
int fa[N];
int dep[N]={0};
bool islight[N]={0};
multiset<int> s[N];
int ctt=0;
int be[N];
void addege(int x,int y){
    cnt++;
    tal[cnt]=y;
    nxt[cnt]=hed[x];
    hed[x]=cnt;
}
void dfs1(int u,int fat){
    size[u]=1;
    fa[u]=fat;
    for(int i=hed[u];i;i=nxt[i]){
        int v=tal[i];
        if(v==fat) continue;
        dep[v]=dep[u]+1;
        dfs1(v,u);
        size[u]+=size[v];
        if(size[v]>size[son[u]]){
            son[u]=v;
        }
    }    
}
void dfs2(int u,int fat,int top){
    tp[u]=top;
    indexy++;
    dfn[u]=indexy;
    id[dfn[u]]=u;
    if(son[u]){
        be[son[u]]=be[u];
        dfs2(son[u],u,top);
    }
    for(int i=hed[u];i;i=nxt[i]){
        int v=tal[i];
        if(v==fat) continue;
        if(v==son[u]) continue;
        be[v]=++ctt;
        dfs2(v,u,v);
    }
}
void jumpy(int POS,int VAL,bool IN){
    if(IN==0){
        multiset<int>::iterator it=s[be[POS]].find(VAL);
        s[be[POS]].erase(it);
    }
    else{
        s[be[POS]].insert(VAL);
    }
}
int midium(int pos){
    int ans=2147483647;
    int u=pos;
    while(u){
        int p=*s[be[u]].begin();
        if(!s[be[u]].empty()){
            if(dep[id[p]]<=dep[u]){
                ans=id[p];
            }
        }
        
        u=fa[tp[u]];
    }
    if(ans==2147483647) ans=-1;
    return ans;
}
int main(){
    scanf("%d%d",&n,&q);
    for(int i=1;i<n;i++){
        int x,y;
        scanf("%d%d",&x,&y);
        addege(x,y);
        addege(y,x);
    }
    dep[0]=0;
    dep[1]=1;
    dfs1(1,1);
    ctt++;
    be[1]=1;
    dfs2(1,1,0);
    for(int i=1;i<=q;i++){
        int opt,pos;
        scanf("%d%d",&opt,&pos);
        if(opt==0){
            islight[dfn[pos]]^=1;
            if(islight[dfn[pos]]==0){jumpy(pos,dfn[pos],0);}else{jumpy(pos,dfn[pos],1);}
        }
        else{
            printf("%d\n",midium(pos));
        }
    }
    return 0;
}

 

posted @ 2019-11-23 22:28  QYJ060604  阅读(120)  评论(0编辑  收藏  举报