洛谷p4197 [克魯斯卡爾樹+主席樹+倍增]

傳送門

題意:給出一個帶權無向圖,圖的邊有權值同時點也有權值,有q次詢問,每次詢問v x k,求從v出發只經過權值不超過x的邊可以走到的權值第k大的點,不存在則輸出-1

題解:由經過不超過x的邊可知可以直接建立克魯斯卡爾樹,同時dfs給原結點重新編號,並得出每個新建結點可以管轄的老節點的區間[L,R],那麼可以對重新編號後的原結點建立主席樹,對每次詢問,倍增得到不超過x的最高結點,同時得到它管轄的區間[L,R],直接求root[R]-root[L-1]的第k大即可

#include<bits/stdc++.h>

using namespace std;
//#define debug(x) cout<<#x<<" is "<<x<<endl;
typedef long long ll;

const int maxn=1e5+5;
const int inf=1e7;

struct edge{
    int fr;
    int to;
    int val;
}e[maxn*5];

struct edge2{
    int fr;
    int to;
    int nex;
}e2[maxn*11];

struct Node{
    int lson;
    int rson;
    int val;
}nod[maxn*21];

int Tot,TOT,cnt,root[maxn],bb[maxn],L[maxn*3],R[maxn*3],val[maxn],id[maxn],head[maxn*3],Val[maxn*3],fa[maxn*3],fa2[31][maxn*3];

bool cmp(struct edge aa,struct edge bb){
    return aa.val<bb.val;
}

void pushup(int rt){
    nod[rt].val=nod[nod[rt].lson].val+nod[nod[rt].rson].val;
}

void build(int &rt,int l,int r){
    rt=++TOT;
    nod[rt].val=0;
    if(l==r)return;
    int mid=(l+r)>>1;
    build(nod[rt].lson,l,mid);
    build(nod[rt].rson,mid+1,r);
}

void update(int &rt,int pre,int l,int r,int v){
    rt=++TOT;
    nod[rt].lson=nod[pre].lson;
    nod[rt].rson=nod[pre].rson;
    nod[rt].val=nod[pre].val;
    if(l==r){nod[rt].val++;return;}
    int mid=(l+r)>>1;
    if(mid>=v)update(nod[rt].lson,nod[pre].lson,l,mid,v);
    else update(nod[rt].rson,nod[pre].rson,mid+1,r,v);
    pushup(rt);
}

void adde(int x,int y){
    e2[cnt].fr=x;
    e2[cnt].to=y;
    e2[cnt].nex=head[x];
    head[x]=cnt++;
}

int finds(int x){
    int xx=x;
    while(fa[x]!=x)x=fa[x];
    while(fa[xx]!=x){
        int t=fa[xx];
        fa[xx]=x;
        xx=t;
    }
    return x;
}

void dfs(int u,int f){
    fa2[0][u]=f;
    for(int i=1;i<=30;i++){
        if(fa2[i-1][u])fa2[i][u]=fa2[i-1][fa2[i-1][u]];
        else break;
    }
    int F=1;
    int L0=1e7;
    int R0=0;
    for(int i=head[u];i!=-1;i=e2[i].nex){
        int v=e2[i].to;
        F=0;
        dfs(v,u);
        L0=min(L0,L[v]);
        R0=max(R0,R[v]);
    }
    if(F){
        L[u]=R[u]=++Tot;
        id[Tot]=u;
    }
    else{
        L[u]=L0;
        R[u]=R0;
    }
}

int query(int rt1,int rt2,int l,int r,int k){
    if(l==r){
        return bb[l];
    }
    int mid=(l+r)>>1;
    if(nod[nod[rt1].lson].val-nod[nod[rt2].lson].val>=k)
        return query(nod[rt1].lson,nod[rt2].lson,l,mid,k);
    else
        return query(nod[rt1].rson,nod[rt2].rson,mid+1,r,k-(nod[nod[rt1].lson].val-nod[nod[rt2].lson].val));
}

int main(){
    int n,m,q;
    memset(head,-1,sizeof(head));
    scanf("%d%d%d",&n,&m,&q);
    for(int i=1;i<=n;i++){scanf("%d",&val[i]);bb[i]=val[i];}
    for(int i=1;i<=3*n;i++)fa[i]=i;
    int tot=0;
    for(int i=1;i<=m;i++){
        int a,b,c;
        scanf("%d%d%d",&a,&b,&c);
        e[++tot].fr=a;
        e[tot].to=b;
        e[tot].val=c;
    }
    sort(e+1,e+1+tot,cmp);
    sort(bb+1,bb+1+n);
    int siz=unique(bb+1,bb+1+n)-(bb+1);
    int num=n;
    for(int i=1;i<=tot;i++){
        int f1=finds(e[i].fr);
        int f2=finds(e[i].to);
        if(f1!=f2){
            num++;
            fa[f1]=num;
            fa[f2]=num;
            adde(num,f1);
            adde(num,f2);
            Val[num]=e[i].val;
        }
    }
    dfs(num,0);
    build(root[0],1,siz);
    for(int i=1;i<=Tot;i++){
        int ww=lower_bound(bb+1,bb+1+siz,val[id[i]])-bb;
        update(root[i],root[i-1],1,siz,ww);
    }
    while(q--){
        int v,x,k;
        scanf("%d%d%d",&v,&x,&k);
        int w=v;
        for(int i=30;i>=0;i--){
            if(Val[fa2[i][w]]<=x&&fa2[i][w]){
                w=fa2[i][w];
            }
        }
        if(R[w]-L[w]+1<k){
            printf("-1\n");
            continue;
        }
        printf("%d\n",query(root[R[w]],root[L[w]-1],1,siz,R[w]-L[w]+2-k));
    }
    return 0;
}

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章