zoukankan      html  css  js  c++  java
  • [题解] LuoguP3302 [SDOI2013]森林

    建议先写一下这个qwq:P2633

    那么这道题的区别就是有了连边操作,还有维护的是森林(这个其实问题不大)。

    再讲一下做法吧...

    假设我现在要查询((u,v))路径上第(k)小的权值,令(T[u])表示点(u)到根的路径上的权值所形成的一棵权值线段树,那么((u,v))路径上的权值线段树就是(T[u]+T[v]-T[lca(u,v)]-T[fa_{lca(u,v)}]),在权值线段树上二分向下走即可。

    现在有了连边操作...咋办?

    大力启发式合并就好了,如果(u)所在的那块的大小比(v)所在那块的大小大的话就让(v)(u)的儿子,否则就让(v)(u)的爸爸。

    发现这样能很方便的维护(u)上面的(2^i)层祖先,倍增算(LCA)就好了。

    另外...那个testcase是数据编号......不是数据的组数......并没有多测......

    (Code:)

    #include <bits/stdc++.h>
    using namespace std;
    #define rep(i,a,n) for (int i=a;i<n;++i)
    #define per(i,a,n) for (int i=n-1;i>=a;--i)
    #define mp make_pair
    #define pb push_back
    #define fi first
    #define se second
    #define all(x) (x).begin(),(x).end()
    #define SZ(x) ((int)(x).size())
    typedef double db;
    typedef long long ll;
    typedef pair<int,int> PII;
    typedef vector<int> VI;
    
    const int N=3e5+10;
    namespace Desc {
        int val[N],tot;
        inline void init(int *a,int n) {
            rep(i,1,n+1) val[i]=a[i];
            sort(val+1,val+n+1);
            tot=unique(val+1,val+n+1)-val-1;
        }
        inline int getid(int x) {return lower_bound(val+1,val+tot+1,x)-val;}
        inline int getval(int i) {return val[i];}
    } // 离散化
    using Desc::getid;
    using Desc::getval;
    
    int lc[N*40],rc[N*40],sum[N*40],oc=0,rt[N];
    inline void clr(int &x) {lc[x]=rc[x]=sum[x]=0;}
    inline int newnode() {int x=++oc;clr(x);return x;}
    
    void build(int &x,int l,int r) {
        x=newnode(); if(l==r) return; int mid=(l+r)>>1;
        build(lc[x],l,mid),build(rc[x],mid+1,r);
    }
    
    void ins(int pre,int &x,int l,int r,int p,int v=1) {
        x=newnode(); lc[x]=lc[pre],rc[x]=rc[pre],sum[x]=sum[pre]+v;
        if(l==r) return; int mid=(l+r)>>1;
        if(p<=mid) ins(lc[pre],lc[x],l,mid,p,v);
        else ins(rc[pre],rc[x],mid+1,r,p,v);
    }
    
    int qry(int x,int y,int lca,int flca,int l,int r,int k) {
        if(l==r) return l; 
        int mid=(l+r)>>1,sz=sum[lc[x]]+sum[lc[y]]-sum[lc[lca]]-sum[lc[flca]];
        if(k<=sz) return qry(lc[x],lc[y],lc[lca],lc[flca],l,mid,k);
        else return qry(rc[x],rc[y],rc[lca],rc[flca],mid+1,r,k-sz);
    }
    
    int n,m,a[N],Q,p[N][30],dep[N];
    VI g[N];
    #define LOGN 21
    
    inline int getlca(int x,int y) {
        if(dep[x]<dep[y]) swap(x,y);
        per(i,0,LOGN) if(dep[p[x][i]]>=dep[y]) x=p[x][i];
        if(x==y) return x;
        per(i,0,LOGN) if(p[x][i]!=p[y][i]) x=p[x][i],y=p[y][i];
        return p[x][0];
    }
    
    int F[N],siz[N],vis[N];
    inline int find(int x) {return F[x]==x?F[x]:F[x]=find(F[x]);}
    
    void dfs(int u,int ff,int deep,int pp) {
        siz[pp]++,vis[u]=1,F[u]=pp;
        int pos=getid(a[u]); ins(rt[ff],rt[u],1,Desc::tot,pos);
        p[u][0]=ff; rep(i,1,LOGN) p[u][i]=p[p[u][i-1]][i-1]; // 倍增LCA
        dep[u]=deep;
        for (auto v:g[u]) if(v!=ff) dfs(v,u,deep+1,pp);
    }
    
    inline void merge(int u,int v) {
        int pu=find(u),pv=find(v);
        if(pu==pv) return;
        if(siz[pu]<siz[pv]) swap(u,v),swap(pu,pv);
        g[u].pb(v),g[v].pb(u);
        dfs(v,u,dep[u]+1,pu);
    }
    
    inline int query(int u,int v,int k) {
        int lca=getlca(u,v),flca=p[lca][0];
        return getval(qry(rt[u],rt[v],rt[lca],rt[flca],1,Desc::tot,k));
    }
    
    inline int nc() {char ch;while(isspace(ch=getchar()));return ch;}
    
    int main() {
    #ifdef LOCAL
        freopen("a.in","r",stdin);
    #endif
        scanf("%*d%d%d%d",&n,&m,&Q);
        rep(i,1,n+1) scanf("%d",a+i);
        Desc::init(a,n);
        build(rt[0],1,Desc::tot);
        rep(i,1,m+1) {
            int u,v; scanf("%d%d",&u,&v);
            g[u].pb(v),g[v].pb(u);
        }
        rep(i,1,n+1) F[i]=i;
        rep(i,1,n+1) if(!vis[i]) dfs(i,0,1,i);
        int lstans=0; while(Q--) {
            char opt=nc();
            int u,v,k;scanf("%d%d",&u,&v);
            if(opt=='Q') {
                scanf("%d",&k);
                u^=lstans,v^=lstans,k^=lstans;
                printf("%d
    ",lstans=query(u,v,k));
            } else {
                u^=lstans,v^=lstans;
                merge(u,v);
            }
        }
        return 0;
    }
    
  • 相关阅读:
    mysql 中文字段排序( 按拼音首字母排序) 的查询语句
    纯css3样式属性制作各种图形图标
    10个超有用的网页设计工具和资源
    手风琴导航效果实现
    css3动画导航实现
    java实现将资源文件转化成sql语句导入数据库
    select实现输入模糊匹配与选择双重功能
    js一些问题总结
    java实现excel与mysql的导入导出
    《C++程序设计》朝花夕拾
  • 原文地址:https://www.cnblogs.com/wxq1229/p/12436925.html
Copyright © 2011-2022 走看看