zoukankan      html  css  js  c++  java
  • [bzoj3673/3674可持久化并查集加强版]

    n个集合 m个操作 操作: 1 a b 合并a,b所在集合 2 k 回到第k次操作之后的状态(查询算作操作) 3 a b 询问a,b是否属于同一集合,是则输出1否则输出0

    0<n,m<=2*10^5  强制在线。

    这两题一题都一样,另一题比较水,nm只有2*10^4,允许离线.....

    做法很简单,把数组当作可持久化线段树那么维护,每个表示区间的节点都不存东西,每次只要新建log个节点。

    我交水的那道过不去,绝望的时候我交了一发加强版居然A了,根据我多年的经验一定是有特殊数据的坑,特判了一波终于过了。

    用了启发式合并之后复杂度nlog^2n

    #include<iostream>
    #include<cstdio>
    #define MN 20000000
    #define MM 200000
    using namespace std;
    inline int read()
    {
        int x = 0 , f = 1; char ch = getchar();
        while(ch < '0' || ch > '9'){ if(ch == '-') f = -1;  ch = getchar();}
        while(ch >= '0' && ch <= '9'){x = x * 10 + ch - '0';ch = getchar();}
        return x * f;
    }
      
    int cnt=0,n,m,last=0,rt[MM+5],cc;
    struct data{
        int x,size;
    }s[MM*3+5];
    struct TREE{
        int l,r;
        data *x;
    }T[MN];
      
    void build(int x,int l,int r)
    {
        if(l==r){T[x].x=&s[l];return;}
        int mid=l+r>>1;
        build(T[x].l=(++cnt),l,mid);
        build(T[x].r=(++cnt),mid+1,r);
    }
      
    data*get(int x,int k,int l=1,int r=n)
    { 
        if(l==r)return T[x].x;
        int mid=l+r>>1;
        if(k<=mid) return get(T[x].l,k,l,mid);
        else return get(T[x].r,k,mid+1,r);  
    }
      
    data getfa(int x,int r)
    {
        data y=*get(r,x),ans=y;
        if(!y.x)return (data){x,ans.size};
        while(y.x) {ans=y;y=*get(r,y.x);}
        return (data){ans.x,y.size};
    }
      
    void ins(int x,int dep,int k)
    {
        int l=1,r=n;int nx=rt[dep]=++cnt;
        while(l<r)
        {
            int mid=l+r>>1;
            if(k<=mid)
            {
                T[nx].r=T[x].r;T[nx].l=++cnt;
                nx=T[nx].l;x=T[x].l;r=mid;
            }
            else
            {
                T[nx].l=T[x].l;T[nx].r=++cnt;
                nx=T[nx].r;x=T[x].r;l=mid+1;
            }
        }
        T[nx].x=&s[cc];
    }
      
    int main()
    {
        cc=n=read();m=read();
        for(int i=1;i<=n;i++)s[i]=(data){0,1};
        build(++cnt,1,n);rt[0]=1;
        for(int i=1;i<=m;i++)
        {
            int a=read(),b=read()^last;
            if(a==2)
                rt[i]=rt[b];
            else
            {
                int c=read()^last;
                if(a==3) printf("%d
    ",last=(getfa(b,rt[i-1]).x==getfa(c,rt[i-1]).x)),rt[i]=rt[i-1];
                else
                {
                    data x=getfa(b,rt[i-1]),y=getfa(c,rt[i-1]);
                    if(x.x==y.x){rt[i]=rt[i-1];continue;}
                    if(x.size>y.size)swap(x,y);
                    s[++cc]=(data){y.x,x.size};ins(rt[i-1],i,x.x);
                    s[++cc]=(data){0,x.size+y.size};ins(rt[i],i,y.x);
                }
            }
        }
        return 0;
    }
  • 相关阅读:
    PHP for循环的写法和示例
    PHP Socket(套接字连接)扩展简介和使用方法
    PHP exec()函数的介绍和使用DEMO
    PHP trim()函数的作用和使用方法
    Linux环境安装xmapp(PHP-Mysql集成环境)
    HP数组转JSON函数json_encode和JSON转数组json_decode函数的使用方法
    PHP 中使用explode()函数切割字符串为数组
    PHP获取随机数的函数rand()和mt_rand()
    PHP stripos()、strripos()和strrpos() 使用方法和区别
    常用工具说明--搭建基于rietveld的CodeReview平台(未测试)
  • 原文地址:https://www.cnblogs.com/FallDream/p/bzoj3674.html
Copyright © 2011-2022 走看看