zoukankan      html  css  js  c++  java
  • 列队[noip2017]

    链接

    其实从我开始学splay后,我就想过开n棵spaly硬钢这道题

    后来我又想到动态开点

    可惜我不会

    终于今天,嗯,我A了它,真好

    #include<bits/stdc++.h>
    #define re return
    #define LL long long
    #define inc(i,l,r) for(LL i=l;i<=r;++i)
    
    using namespace std;
    template<typename T>inline void rd(T&x)
    {
        char c;bool f=0;
        while((c=getchar())<'0'||c>'9')if(c=='-')f=1;
        x=c^48;
        while((c=getchar())>='0'&&c<='9')x=x*10+(c^48);
        if(f)x=-x;
    }
    
    const int maxn=3000000;
    LL n,m,tot,cnt,Q;
    int fa[maxn],son[maxn][2];
    LL size[maxn],ll[maxn],rr[maxn];
    
    /*此题区间左闭右开*/
    struct Splay
    {
        int rt;
        //建新点 
        inline int New_point(LL l,LL r)
        {
            ++cnt;
            fa[cnt]=son[cnt][0]=son[cnt][1]=0;
            ll[cnt]=l;rr[cnt]=r;size[cnt]=r-l;
            re cnt;
        }
        //saply常规操作
        inline int chk(int x){re son[fa[x]][1]==x;}
        inline void pushup(int x)
        {
            size[x]=size[son[x][0]]+size[son[x][1]]+rr[x]-ll[x];
        } 
        
        inline void rotate(int x)
        {
            int y=fa[x],z=fa[y],f=chk(x),w=son[x][f^1];
            son[z][chk(y)]=x;fa[x]=z;
            son[x][f^1]=y;fa[y]=x;
            son[y][f]=w;fa[w]=y;
            pushup(y);
            pushup(x);
        }
        
        inline void splay(int x,int goal=0)
        {
            while(fa[x]!=goal)
            {
                int y=fa[x],z=fa[y];
                if(z!=goal)
                chk(x)==chk(y)?rotate(y):rotate(x);
                rotate(x);
            }
            if(!goal)rt=x;
        }
        
        //动态开点平衡树 (将n多操作合二为一) 
    
        //你从未见过的船新操作,动态开点
        //将前k个点放入原根,后面新开节点 
        inline LL Split_point(int u,LL k)
        {
            k+=ll[u];
            int y=New_point(k,rr[u]);
            rr[u]=k;
            if(!son[u][1])
                fa[son[u][1]=y]=u;
            else 
            {
                int t=son[u][1];
                while(son[t][0])t=son[t][0];
                fa[son[t][0]=y]=t;
                while(t!=u)pushup(t),t=fa[t];
            }
            splay(y);
            re y;
        } 
        
        //找出第K大点并弹出 
         LL Pop_kth(LL k)
        {
            int u=rt;
            while(2333)
            {
                int y=son[u][0];
                if(size[y]>=k)u=y;
                else 
                {
                    k-=size[y];
                    if(rr[u]-ll[u]<k)k=k-(rr[u]-ll[u]),u=son[u][1];
                    else 
                    {
                        if(k!=rr[u]-ll[u])Split_point(u,k);
                        if(k!=1)u=Split_point(u,k-1);
                        break;
                    } 
                }
            }
            
            splay(u);
            fa[son[u][0]]=fa[son[u][1]]=0;
            if(!son[u][0])
            {
                rt=son[u][1];
            }
            else
            {
                int t=son[u][0];
                while(son[t][1])t=son[t][1];
                splay(t);
                rt=t;
                fa[son[t][1]=son[u][1]]=t;
                pushup(t);
            }
            re ll[u]; 
        }
        
        
        //找到最后,并放进新数 
        inline void Push_back(LL x)
        {
            int p=New_point(x,x+1);
            if(!rt)    rt=p;
            else
            {
                int u=rt;
                while(son[u][1])
                    u=son[u][1];
                splay(u); 
                son[fa[p]=u][1]=p;
                pushup(u);
            }
        }
        
    }s[maxn];
    
    int main()
    {
        //freopen("in.txt","r",stdin);
        freopen("testdata.in","r",stdin);
        LL x,y,p;
        
        rd(n),rd(m),rd(Q);
        
        //建树 
        inc(i,1,n) s[i].rt=s[i].New_point((i-1)*m+1,i*m);
        s[0].rt=s[0].New_point(m,m+1);
        inc(i,2,n)
            s[0].Push_back(i*m);
        
        inc(i,1,Q)
        {
            rd(x),rd(y);
            s[x].Push_back(s[0].Pop_kth(x));
            printf("%lld
    ",p=s[x].Pop_kth(y));    
            s[0].Push_back(p);
        }  
        
        re 0;
    } 

    当然,动态开点权值线段树什么的最适合这道题了

    /*
    魔改程序伤不起
    LL落下两行泪 
    */
    #include<bits/stdc++.h>
    #define re return
    #define ll long long
    #define inc(i,l,r) for(ll i=l;i<=r;++i)
    
    using namespace std;
    template<typename T>inline void rd(T&x)
    {
        char c;bool f=0;
        while((c=getchar())<'0'||c>'9')if(c=='-')f=1;
        x=c^48;
        while((c=getchar())>='0'&&c<='9')x=x*10+(c^48);
        if(f)x=-x;
    }
    
    const int maxn=7000000;
    ll n,m,tot,cnt,Q,flag,ans;
    ll x,y;
    ll use[maxn],ls[maxn],rs[maxn],num[maxn];
    ll sum[300005],T[300005];
    
    inline void query(ll &rt,ll l,ll r,ll k)
    {
        if(!rt)rt=++cnt;
        //没有访问过 开点 
        ++use[rt];
        //因为下面用的是use[ls[rt]],没有影响 
        if(l==r)
        {
            if(num[rt])ans=num[rt];
            else if(!flag)ans=l*m;
            else ans=(x-1)*m+l;
            re ;
        }
        
        ll mid=(l+r)>>1;
        ll cur=mid-l+1-use[ls[rt]];
        //标记真实存在的个数 
        if(cur>=k)query(ls[rt],l,mid,k);
        else query(rs[rt],mid+1,r,k-cur);
        re ;
    }
    
    inline void insert(ll &rt,ll l,ll r,ll pos)
    {
        if(!rt)rt=++cnt;
        //没有访问过 开点 
        if(l==r)
        { 
            num[rt]=ans;
            re;
        }    
        
        ll mid=(l+r)>>1;
        if(pos<=mid)insert(ls[rt],l,mid,pos);
        else insert(rs[rt],mid+1,r,pos);
    }
    int main()
    {
        //freopen("in.txt","r",stdin);
        freopen("testdata.in","r",stdin);
        ll p,t;
        
        rd(n),rd(m),rd(Q);
        inc(i,1,Q)    
        {
            rd(x),rd(y);
            if(y==m)
            {
                flag=0;
                query(T[n+1],1,n+Q,x);
                printf("%lld
    ",ans);
                ++sum[n+1];//现在放入位置+n
                insert(T[n+1],1,n+Q,sum[n+1]+n);
           //预留好足够大的位置
    //加入末尾 } else { flag=x; query(T[x],1,m+Q,y); printf("%lld ",ans); ++sum[n+1]; insert(T[n+1],1,n+Q,sum[n+1]+n); flag=0; query(T[n+1],1,n+Q,x); ++sum[x]; insert(T[x],1,m+Q,sum[x]+m-1); //此处之所以,是因为每一行的m列不被包括在那一行中
          //将每一行中的m抹去
    } } re 0; }

     我太蒟蒻了,还没理解树状数组的做法,待我有时

  • 相关阅读:
    fzu 2122
    hdu 4707 bellman
    sicily 10330. Cutting Sausages
    湖南省2016省赛题。1809: Parenthesis 线段树
    Panoramic Photography
    B. No Time for Dragons 贪心
    The Weakest Sith
    E. The Best among Equals
    Gym 101149I I
    AtCoder D
  • 原文地址:https://www.cnblogs.com/lsyyy/p/11396529.html
Copyright © 2011-2022 走看看