【题目分析】

听说是树套树。(雾)

怒写树状数组套主席树,然后就Rank1了。23333

单点修改,区间查询+k大数查询=树状数组套主席树。

【代码】

#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <cmath>

#include <set>
#include <map>
#include <string>
#include <algorithm>
#include <vector>
#include <iostream>
#include <queue>
using namespace std;

#define maxn 50005
#define mlog 20

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 n,m;
int rt[maxn<<1],a[maxn],b[maxn<<1],cnt=0,tot=0,sum,rnk,TMP;
int ls[maxn<<6],rs[maxn<<6],siz[maxn<<6];
int L[maxn],R[maxn],opt[maxn],x[maxn],y[maxn],z[maxn];

void ins(int o1,int & o2,int l,int r,int x,int f)
{
//    printf("ins %d %d %d %d %d %d\n",o1,o2,l,r,x,f);
    o2=++tot;
    siz[o2]=siz[o1]+f;
    if (l==r) return ;
    int mid=(l+r)/2;
    if (x<=mid) rs[o2]=rs[o1],ins(ls[o1],ls[o2],l,mid,x,f);
    else ls[o2]=ls[o1],ins(rs[o1],rs[o2],mid+1,r,x,f);
    return ;
}

void ready(int l,int r)
{
//  cout<<"ready for "<<l<<" "<<r<<endl;
    l--;
    L[0]=R[0]=0;
    for (int j=l;j;j-=j&(-j)) L[++L[0]]=rt[j];
    for (int j=r;j;j-=j&(-j)) R[++R[0]]=rt[j];
//  cout<<"L "; for (int j=1;j<=L[0];++j) cout<<L[j]<<" "; cout<<endl;
//  cout<<"R "; for (int j=1;j<=R[0];++j) cout<<R[j]<<" "; cout<<endl;
//  cout<<"L "; for (int j=1;j<=L[0];++j) cout<<siz[L[j]]<<" "; cout<<endl;
//  cout<<"R "; for (int j=1;j<=R[0];++j) cout<<siz[R[j]]<<" "; cout<<endl;
}

void cholef()
{
//  cout<<"choose left"<<endl;
    for (int i=1;i<=L[0];++i) L[i]=ls[L[i]];
    for (int i=1;i<=R[0];++i) R[i]=ls[R[i]];
}

void chorig()
{
//  cout<<"choose right"<<endl;
    for (int i=1;i<=L[0];++i) L[i]=rs[L[i]];
    for (int i=1;i<=R[0];++i) R[i]=rs[R[i]];
}

int taksum()
{
    int ret=0;
    for (int i=1;i<=L[0];++i) ret-=siz[ls[L[i]]];
    for (int i=1;i<=R[0];++i) ret+=siz[ls[R[i]]];
    return ret;
}

int qrnk(int l,int r,int x)
{
//  printf("qrnk %d %d %d\n",l,r,x);
    if (l==r) return 1;
    int mid=(l+r)/2,tmp=taksum();
//  cout<<"take sum "<<tmp<<endl;
    if (x<=mid)  return cholef(),qrnk(l,mid,x);
    else return chorig(),tmp+qrnk(mid+1,r,x);
}

int qnum(int l,int r,int x)
{
    if (l==r) return l;
    int mid=(l+r)/2,tmp=taksum();
    if (x<=tmp) return cholef(),qnum(l,mid,x);
    else return chorig(),qnum(mid+1,r,x-tmp);
}

int main()
{
    n=read();m=read();
    for (int i=1;i<=n;++i) scanf("%d",&a[i]),b[++cnt]=a[i];
    for (int i=1;i<=m;++i)
    {
        scanf("%d%d%d",&opt[i],&x[i],&y[i]);
        if (opt[i]!=3) scanf("%d",&z[i]);
        if (opt[i]==3) b[++cnt]=y[i];
        if (opt[i]==4||opt[i]==5) b[++cnt]=z[i];
    }
    sort(b+1,b+cnt+1);
    cnt=unique(b+1,b+cnt+1)-b-1;
    for (int i=1;i<=n;++i) a[i]=lower_bound(b+1,b+cnt+1,a[i])-b;
    for (int i=1;i<=m;++i)
    {
        if (opt[i]==1||opt[i]==4||opt[i]==5)z[i]=lower_bound(b+1,b+cnt+1,z[i])-b;
        if (opt[i]==3) y[i]=lower_bound(b+1,b+cnt+1,y[i])-b;
    }
//  cout<<endl<<endl;
//  for (int i=1;i<=cnt;++i) cout<<i<<" ";cout<<endl;
//  for (int i=1;i<=cnt;++i) cout<<b[i]<<" "; cout<<endl;
//  for (int i=1;i<=m;++i) cout<<opt[i]<<" "<<x[i]<<" "<<y[i]<<" "<<z[i]<<endl;
//  cout<<endl<<endl;
    for (int i=1;i<=n;++i)
        for (int j=i;j<=n;j+=j&(-j))
            ins(rt[j],rt[j],1,cnt,a[i],1);
    for (int i=1;i<=m;++i)
    {
        switch(opt[i])
        {
            case 1:
                ready(x[i],y[i]);
                printf("%d\n",qrnk(1,cnt,z[i]));
            break;
            case 3:
                for (int j=x[i];j<=n;j+=j&(-j))
                {
                    ins(rt[j],rt[j],1,cnt,a[x[i]],-1);
                    ins(rt[j],rt[j],1,cnt,y[i],1);
                }
                a[x[i]]=y[i];
            break;
            case 2:
                ready(x[i],y[i]);
                printf("%d\n",b[qnum(1,cnt,z[i])]);
            break;
            case 4:
                ready(x[i],y[i]);
                rnk=qrnk(1,cnt,z[i]);
//              printf("rnk is %d\n",rnk);
                ready(x[i],y[i]);
                printf("%d\n",b[qnum(1,cnt,rnk-1)]);
            break;
            case 5:
                ready(x[i],y[i]);
                TMP=0;
                TMP-=qrnk(1,cnt,z[i]);
                ready(x[i],y[i]);
                TMP+=qrnk(1,cnt,z[i]+1);
//              printf("it s have %d\n",TMP);
                ready(x[i],y[i]);
                rnk=qrnk(1,cnt,z[i]);
                ready(x[i],y[i]);
                printf("%d\n",b[qnum(1,cnt,rnk+TMP)]);

        }
    }
}

  

BZOJ 3196 Tyvj 1730 二逼平衡树 ——树状数组套主席树的更多相关文章

  1. bzoj 3196 Tyvj 1730 二逼平衡树(线段树套名次树)

    3196: Tyvj 1730 二逼平衡树 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1807  Solved: 772[Submit][Stat ...

  2. BZOJ 3196: Tyvj 1730 二逼平衡树( 树套树 )

    这道题做法应该很多吧.... 我用了线段树套treap.... -------------------------------------------------------------------- ...

  3. bzoj 3196/ Tyvj 1730 二逼平衡树 (线段树套平衡树)

    3196: Tyvj 1730 二逼平衡树 Time Limit: 10 Sec  Memory Limit: 128 MB[Submit][Status][Discuss] Description ...

  4. BZOJ - 3196 Tyvj 1730 二逼平衡树 (线段树套treap)

    题目链接 区间线段树套treap,空间复杂度$O(nlogn)$,时间复杂度除了查询区间k大是$O(log^3n)$以外都是$O(log^2n)$的. (据说线段树套线段树.树状数组套线段树也能过?) ...

  5. 【BZOJ】3196: Tyvj 1730 二逼平衡树(区间第k小+树套树)

    http://www.lydsy.com/JudgeOnline/problem.php?id=3196 Treap+树状数组 1WA1A,好伤心,本来是可以直接1A的,这次开始我并没有看题解,就写出 ...

  6. bzoj 3196: Tyvj 1730 二逼平衡树

    #include<cstdio> #include<ctime> #include<cstdlib> #include<iostream> #defin ...

  7. BZOJ 3196 Tyvj 1730 二逼平衡树 树套树 线段树 treap

    http://www.lydsy.com/JudgeOnline/problem.php?id=3196 http://hzwer.com/2734.html 线段树套treap,似乎splay也可以 ...

  8. BZOJ 3196 Tyvj 1730 二逼平衡树:线段树套splay

    传送门 题意 给你一个长度为 $ n $ 有序数列 $ a $ ,进行 $ m $ 次操作,操作有如下几种: 查询 $ k $ 在区间 $ [l,r] $ 内的排名 查询区间 $ [l,r] $ 内排 ...

  9. bzoj 3196 Tyvj 1730 二逼平衡树【线段树 套 splay】

    四舍五入就是个暴力. 对于线段树的每个区间都开一棵按权值排序的splay 对于第二个操作,二分一下,每次查询mid的排名,复杂度 $ O(nlog(n)^{3}) $ 其余的操作都是$ O(nlog( ...

随机推荐

  1. Brackets前端开发IDE工具

    Brackets是一个开源的前端开发IDE工具,网页设计师和前端开发人员必备的前端开发IDE工具. 它能够使你在开发WEB网站实时预览你的网页,目前版本只适用于Chrome浏览器可以实时预览效果 支持 ...

  2. Arcgis 10.1安装

    来源http://download.csdn.net/detail/l_j_kin/7310665#comment 测试有效,存着备用,转过来不自己写了 安装desktop 打开“注册机”,功能选择a ...

  3. WampServer 的phpmyadmin数据

    WampServer首次安装的时候phpmyadmin的密码是为空 设置密码 1.安装成功后,通过 phpmyadmin 进入mysql,点击上面的 [用户] 菜单,在用户[root]主机[local ...

  4. CPU

    多核处理器 http://baike.baidu.com/link?url=6LwImqyaZqI15gVqcGstOA5S73g-Gj2hakrCbFGc_Jh1NIPPZLkahpuI5OSLoi ...

  5. 转载:Android调用相册、拍照实现缩放、切割图片

    好几天没有写博客了,感觉都有点懈怠了.笔者参加了大学生第二届软件设计大赛,这几天 一直在弄大赛的事情,没有花些时间来整理博客.好在经过一些时日比赛的东西也弄得差不多了, 接下来就是将这段时间学习里面有 ...

  6. Java 8之二小坑:stream parallel 和 lamada

    Stream:parallel乱序 Java 8 stream流为处理集合时非常方便.遇到的一个坑是为了提高在多核cpu下的性能,尝试了parallel().数据源是HashSet的,在做分割的时候发 ...

  7. FreeImage编译及遇到问题解决

    FreeImage编译及遇到问题解决 1.下载freeImage源码包 wget http://downloads.sourceforge.net/freeimage/FreeImage3170.zi ...

  8. django orm字段和参数

    字段 1.models.AutoField 自增列 = int(11) 如果没有的话,默认会生成一个名称为 id 的列,如果要显示的自定义一个自增列,必须将给列设置为主键 primary_key=Tr ...

  9. WSME api controller嵌套使用wtypes

    # 定义user类型和user列表类型 from wsme import types as wtypes class User(wtypes.Base): name = wtypes.text age ...

  10. XSS攻击及防御

    XSS又称CSS,全称Cross SiteScript,跨站脚本攻击,是Web程序中常见的漏洞,XSS属于被动式且用于客户端的攻击方式,所以容易被忽略其危害性.其原理是攻击者向有XSS漏洞的网站中输入 ...