数据结构综合

P1279 文艺平衡树

额真的是平衡树啊…调了0.5h有点方

#include <iostream>
#include <stdio.h>
#include <stdlib.h>
#include <algorithm>
#include <string.h>
#include <vector>
#include <limits>
#include <set>
#include <map>
using namespace std;
#define SZ 666666
int ch[SZ][2],fa[SZ],sz[SZ],val[SZ],an,root;
bool rev[SZ];
void init()
{
    root=1; ch[1][0]=0; ch[1][1]=2; sz[1]=2;
    fa[2]=1; ch[2][0]=ch[2][1]=0; sz[2]=1; an=2;
}
void pd(int x)
{
    if(!rev[x]) return;
    swap(ch[x][0],ch[x][1]);
    rev[ch[x][0]]^=1;
    rev[ch[x][1]]^=1;
    rev[x]=0;
}
void upd(int x)
{
    sz[x]=sz[ch[x][0]]+sz[ch[x][1]]+1;
}
void rot(int x)
{
    int y=fa[x],d=ch[y][0]==x; pd(y); pd(x);
    int f=fa[y]; fa[x]=f; fa[y]=x;
    if(f) ch[f][ch[f][1]==y]=x;
    int p=ch[x][d]; ch[x][d]=y; ch[y][!d]=p;
    if(p) fa[p]=y; upd(y); upd(x); if(y==root) root=x;
}
void splay(int x,int f)
{
    pd(x);
    while(fa[x]!=f)
    {
        int y=fa[x];
        if(fa[y]!=f)
        {
            if(ch[fa[y]][1]==y^ch[y][0]==x) rot(x);
            else rot(y);
        }
        rot(x);
    }
    upd(x);
    if(!f) root=x;
}
void splayp(int x,int f)
{
    int c=root;
    while(pd(c),sz[ch[c][0]]!=x-1)
    {
        if(sz[ch[c][0]]<x-1) x-=sz[ch[c][0]]+1,c=ch[c][1];
        else c=ch[c][0];
    }
    splay(c,f);
}
int n;
#define RRL ch[ch[root][1]][0]
void addnode(int& x,int f,int v)
{
    x=++an; fa[x]=f; sz[x]=1; val[x]=v; ch[x][0]=ch[x][1]=rev[x]=0;
}
void build(int& x,int f,int l,int r)
{
    if(l>r) {x=0; return;}
    int mid=l+r>>1;
    addnode(x,f,mid);
    build(ch[x][0],x,l,mid-1);
    build(ch[x][1],x,mid+1,r);
    upd(x);
}
void revs(int l,int r)
{
    splayp(l,0);
    splayp(r+2,root);
    rev[RRL]^=1;
}
void prt(int cur)
{
    if(!cur) return;
    pd(cur);
    prt(ch[cur][0]);
    if(val[cur]) printf("%d ",val[cur]);
    prt(ch[cur][1]);
}
int main()
{
    int q;
    scanf("%d%d",&n,&q); init();
    build(RRL,ch[root][1],1,n);
    upd(ch[root][1]); upd(root);
    while(q--)
    {
        int l,r;
        scanf("%d%d",&l,&r);
        revs(l,r);
    }
    prt(root);
}

FJOI2016 神秘数

#include <iostream>
#include <stdio.h>
#include <stdlib.h>
#include <algorithm>
#include <string.h>
#include <vector>
#include <limits>
#include <set>
#include <map>
using namespace std;
#define SZ 666666
#define S2 5555555
int ch[S2][2],rot[SZ],sum[S2],an=0;
void ins(int r1,int& r2,int l,int r,int p,int q)
{
    if(!r2) r2=++an;
    sum[r2]=sum[r1]+q;
    if(l==r) return;
    int mid=l+r>>1;
    if(p<=mid) ins(ch[r1][0],ch[r2][0],l,mid,p,q), ch[r2][1]=ch[r1][1];
    else ins(ch[r1][1],ch[r2][1],mid+1,r,p,q), ch[r2][0]=ch[r1][0];
}
int query(int r1,int r2,int l,int r,int p)
{
    if(l>p) return 0;
    if(sum[r1]==sum[r2]) return 0;
    if(r<=p) return sum[r2]-sum[r1];
    int mid=l+r>>1,ans=0;
    ans+=query(ch[r1][0],ch[r2][0],l,mid,min(p,mid));
    if(p>mid) ans+=query(ch[r1][1],ch[r2][1],mid+1,r,p);
    return ans;
}
int n,q,a[SZ],mm=1;
int ans(int l,int r)
{
    int cur=0;
    while(1)
    {
        int cc=query(rot[l-1],rot[r],1,mm,cur+1);
        if(cc==cur) return cc+1;
        cur=cc;
    }
}
int main()
{
    scanf("%d",&n);
    for(int i=1;i<=n;i++) scanf("%d",a+i), mm=max(mm,a[i]);
    for(int i=1;i<=n;i++) ins(rot[i-1],rot[i],1,mm,a[i],a[i]);
    scanf("%d",&q);
    while(q--)
    {
        int l,r;
        scanf("%d%d",&l,&r);
        printf("%d\n",ans(l,r));
    }
}

bzoj3110 K大数查询

#include <iostream>
#include <stdio.h>
#include <stdlib.h>
#include <algorithm>
#include <string.h>
#include <vector>
#include <limits>
#include <set>
#include <map>
using namespace std;
#define SZ 23333333
int an=0,rot[SZ],sum[SZ],tag[SZ],ch[SZ][2],n;
int alc(int& x) {return (x)?(x):(x=++an);}
void I_add(int x,int l,int r,int ql,int qr)
{
    sum[x]+=qr-ql+1;
    if(l==ql&&r==qr) {++tag[x]; return;}
    int mid=l+r>>1;
    if(ql<=min(mid,qr)) I_add(alc(ch[x][0]),l,mid,ql,min(mid,qr));
    if(max(mid+1,ql)<=qr) I_add(alc(ch[x][1]),mid+1,r,max(mid+1,ql),qr);
}
void O_add(int x,int l,int r,int ql,int qr,int p)
{
    I_add(alc(rot[x]),1,n,ql,qr);
    if(l==r) return;
    int mid=l+r>>1;
    if(p<=mid) O_add(x+x,l,mid,ql,qr,p);
    else O_add(x+x+1,mid+1,r,ql,qr,p);
}
int I_sum(int x,int l,int r,int ql,int qr)
{
    if(!x||ql>qr) return 0;
    if(l==ql&&r==qr) return sum[x];
    int mid=l+r>>1;
    return tag[x]*(qr-ql+1)+I_sum(ch[x][0],l,mid,ql,min(qr,mid))+I_sum(ch[x][1],mid+1,r,max(ql,mid+1),qr);
}
int O_query(int x,int l,int r,int ql,int qr,int p)
{
    if(l==r) return l;
    int mid=l+r>>1,cnt=I_sum(rot[x+x+1],1,n,ql,qr);
    if(cnt>=p) return O_query(x+x+1,mid+1,r,ql,qr,p);
    else return O_query(x+x,l,mid,ql,qr,p-cnt);
}
int main()
{
    int q; scanf("%d%d",&n,&q);
    while(q--)
    {
        int t,a,b,c; scanf("%d%d%d%d",&t,&a,&b,&c);
        if(t==1) O_add(1,1,n,a,b,c);
        else printf("%d\n",O_query(1,1,n,a,b,c));
    }
}

时间关系就写到这里…碎觉

posted @ 2016-04-09 23:02  fjzzq2002  阅读(354)  评论(0编辑  收藏  举报