题意

您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作:

  1. 查询k在区间内的排名

  2. 查询区间内排名为k的值

  3. 修改某一位值上的数值

  4. 查询k在区间内的前驱(前驱定义为严格小于x,且最大的数,若不存在输出-2147483647)

  5. 查询k在区间内的后继(后继定义为严格大于x,且最小的数,若不存在输出2147483647)

注意上面两条要求和tyvj或者bzoj不一样,请注意

说明

时空限制:2s,128M

\(n,m \leq 5\cdot {10}^4\)

保证有序序列所有值在任何时刻满足$ [0, {10} ^8]$

分析

模板题。

树状数组套权值线段树,时空复杂度\(O((n+m) \log_2^2n)\)

代码

#include<bits/stdc++.h>
#define rg register
#define il inline
#define co const
#define lowbit(x) (x&-x)
template<class T>il T read(){
    rg T data=0,w=1;
    rg char ch=getchar();
    while(!isdigit(ch)){
        if(ch=='-') w=-1;
        ch=getchar();
    }
    while(isdigit(ch))
        data=data*10+ch-'0',ch=getchar();
    return data*w;
}
template<class T>il T read(rg T&x){
    return x=read<T>();
}
typedef long long ll;
using std::vector;
vector<int> vs;

co int N=5e4+1,LG=120;
int c[N],n,m;
struct quiz{
    int opt,l,r,k;
}q[N];
// Tree Tao Tree
namespace T{
    // Interval Tree
    int L[N*LG],R[N*LG],sum[N*LG];
    int tot,root[N];
    int rank(vector<int>&x,int l,int r,int p){
        if(l==r) return 1;
        int m=(l+r)/2;
        if(p<=m){
            for(int i=0;i<x.size();++i)
                x[i]=x[i]<0?-L[-x[i]]:L[x[i]];
            return rank(x,l,m,p);
        }
        else{
            int s=0;
            for(int i=0;i<x.size();++i){
                x[i]<0?s-=sum[L[-x[i]]]:s+=sum[L[x[i]]];
                x[i]=x[i]<0?-R[-x[i]]:R[x[i]];
            }
            return s+rank(x,m+1,r,p);
        }
    }
    int kth(vector<int>&x,int l,int r,int k){
        if(l==r) return l;
        int s=0;
        for(int i=0;i<x.size();++i)
            x[i]<0?s-=sum[L[-x[i]]]:s+=sum[L[x[i]]];
        int m=(l+r)/2;
        if(s>=k){
            for(int i=0;i<x.size();++i)
                x[i]=x[i]<0?-L[-x[i]]:L[x[i]];
            return kth(x,l,m,k);
        }
        else{
            for(int i=0;i<x.size();++i)
                x[i]=x[i]<0?-R[-x[i]]:R[x[i]];
            return kth(x,m+1,r,k-s);
        }
    }
    void change(int&x,int l,int r,int p,int d){
        if(!x) x=++tot;
        sum[x]+=d;
        if(l==r) return;
        int m=(l+r)/2;
        if(p<=m) change(L[x],l,m,p,d);
        else change(R[x],m+1,r,p,d);
    }
    int precessor(vector<int>&x,int l,int r,int p){
        if(p==1) return -1;
        vector<int>y;
        int k=rank(y=x,l,r,p);
        if(k==1) return -1;
        return kth(y=x,l,r,k-1);
    }
    int successor(vector<int>&x,int l,int r,int p){
        if(p==vs.size()) return -1;
        vector<int>y;
        int k=rank(y=x,l,r,p+1)-1,s=0; // edit 1: reuse rank
        for(int i=0;i<x.size();++i)
            x[i]<0?s-=sum[-x[i]]:s+=sum[x[i]];
        if(k==s) return -1;
        return kth(y=x,l,r,k+1);
    }
    // Binaru Index Tree
    int rank(int l,int r,int v){
        vector<int> roots;
        for(int i=r;i;i-=lowbit(i))
            roots.push_back(root[i]);
        for(int i=l-1;i;i-=lowbit(i))
            roots.push_back(-root[i]);
        return rank(roots,1,vs.size(),v);
    }
    int kth(int l,int r,int k){
        vector<int> roots;
        for(int i=r;i;i-=lowbit(i))
            roots.push_back(root[i]);
        for(int i=l-1;i;i-=lowbit(i))
            roots.push_back(-root[i]);
        return kth(roots,1,vs.size(),k);
    }
    void change(int p,int v){
        for(int i=p;i<=n;i+=lowbit(i)){
            change(root[i],1,vs.size(),c[p],-1); // edit 1:c[p]
            change(root[i],1,vs.size(),v,1);
        }
        c[p]=v;
    }
    int precessor(int l,int r,int v){
        vector<int> roots;
        for(int i=r;i;i-=lowbit(i))
            roots.push_back(root[i]);
        for(int i=l-1;i;i-=lowbit(i))
            roots.push_back(-root[i]);
        return precessor(roots,1,vs.size(),v);
    }
    int successor(int l,int r,int v){
        vector<int> roots;
        for(int i=r;i;i-=lowbit(i))
            roots.push_back(root[i]);
        for(int i=l-1;i;i-=lowbit(i))
            roots.push_back(-root[i]);
        return successor(roots,1,vs.size(),v);
    }
}

int main(){
//  freopen(".in","r",stdin);
//  freopen(".out","w",stdout);
    read(n),read(m);
    for(int i=1;i<=n;++i)
        vs.push_back(read(c[i]));
    for(int i=1;i<=m;++i){
        read(q[i].opt);
        if(q[i].opt!=3){
            read(q[i].l),read(q[i].r),read(q[i].k);
            if(q[i].opt!=2) vs.push_back(q[i].k);
        }
        else read(q[i].l),vs.push_back(read(q[i].k));
    }
    sort(vs.begin(),vs.end()),vs.erase(unique(vs.begin(),vs.end()),vs.end());
    for(int i=1;i<=n;++i){
        c[i]=lower_bound(vs.begin(),vs.end(),c[i])-vs.begin()+1;
        for(int j=i;j<=n;j+=lowbit(j)) // insert
            T::change(T::root[j],1,vs.size(),c[i],1);
    }
    for(int i=1;i<=m;++i){
        if(q[i].opt!=2) q[i].k=lower_bound(vs.begin(),vs.end(),q[i].k)-vs.begin()+1; // edit 2: only operator 3
        if(q[i].opt==1) printf("%d\n",T::rank(q[i].l,q[i].r,q[i].k));
        else if(q[i].opt==2) printf("%d\n",vs[T::kth(q[i].l,q[i].r,q[i].k)-1]);
        else if(q[i].opt==3) T::change(q[i].l,q[i].k);
        else if(q[i].opt==4){
            int re=T::precessor(q[i].l,q[i].r,q[i].k);
            if(re==-1) puts("-2147483647");
            else printf("%d\n",vs[re-1]);
        }
        else if(q[i].opt==5){
            int re=T::successor(q[i].l,q[i].r,q[i].k);
            if(re==-1) puts("2147483647");
            else printf("%d\n",vs[re-1]);
        }
    }
    return 0;
}

LG3380 3380 【模板】二逼平衡树(树套树)的更多相关文章

  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 二逼平衡树 (线段树套平衡树)

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

  3. BZOJ3196 & 洛谷3380:二逼平衡树——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=3196 https://www.luogu.org/problemnew/show/P3380 (题 ...

  4. BZOJ3196 二逼平衡树 ZKW线段树套vector(滑稽)

    我实在是不想再打一遍树状数组套替罪羊树了... 然后在普通平衡树瞎逛的时候找到了以前看过vector题解 于是我想:为啥不把平衡树换成vector呢??? 然后我又去学了一下ZKW线段树 就用ZKW线 ...

  5. BZOJ3196 二逼平衡树 【线段树套平衡树】

    题目 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作: 1.查询k在区间内的排名 2.查询区间内排名为k的值 3.修改某一位值上的数值 4.查询k在区间内的前驱(前驱 ...

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

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

  7. [BZOJ3196] [Tyvj1730] 二逼平衡树(线段树 套 Splay)

    传送门 至少BZOJ过了,其他的直接弃. 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作: 1.查询k在区间内的排名 2.查询区间内排名为k的值 3.修改某一位值上的 ...

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

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

  9. ZJOI2013 K大数查询 和 LG3380【模板】二逼平衡树(树套树)

    K大数查询 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c:如果是2 a b c形式,表示询问从第a个位置到第b个位置,第C大的 ...

  10. 【BZOJ 3196】二逼平衡树 线段树套splay 模板题

    我写的是线段树套splay,网上很多人写的都是套treap,然而本蒟蒻并不会treap 奉上sth神犇的模板: //bzoj3196 二逼平衡树,支持修改某个点的值,查询区间第k小值,查询区间某个值排 ...

随机推荐

  1. python django model filter 条件过滤,及多表连接查询、反向查询,某字段的distinct[转]

    1.多表连接查询:当我知道这点的时候顿时觉得django太NX了.   class A(models.Model):     name = models.CharField(u'名称')   clas ...

  2. C++转C#函数事例

    using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.R ...

  3. java socket - 半关闭

    通常,使用关闭输出流来表示输出已经结束.但在进行网络通信时则不能这样做.因为我们关闭输出流时,该输出流对应的Socket也将随之关闭,这样程序将无法再从该socket中读取数据. 为了应付这种情况,s ...

  4. Gulp实例(包括环境搭建的自动检测)

    # Gulp实例(包括环境搭建的自动检测) Gulp是一个自动化构建工具,开发者可以使用它在项目开发过程中自动执行常见任务.下面我将完成如下的功能点并且附上源码: CSS文件打包 less文件打包 s ...

  5. css类名大全以及其他关键词

    标签 abbr 缩写的标记 token-based-auth-backend  后端 token-based-auth-frontend  前端 signin 登陆  Sign in / Sign u ...

  6. Jexus部署Asp.Net Core项目

    在之前的我的博客项目中,我将.net Core发布到Cent OS 上,使用的Nginx代理以及Supervisor进程守护,看过我的博客的童鞋,也会发现,这种方式比较麻烦,光命令行就看的头大,总共部 ...

  7. python 正则表达式使用

    正则表达式总结: python支持的字符和语法:. 匹配任意除换行符\n 之外的所有字符\ 转义字符 比如说要匹配的字符中含有 \ . * 等可以使用\转义 例如 \* \. \\[.....] 直接 ...

  8. Python 用Redis简单实现分布式爬虫

    Redis通常被认为是一种持久化的存储器关键字-值型存储,可以用于几台机子之间的数据共享平台. 连接数据库 注意:假设现有几台在同一局域网内的机器分别为Master和几个Slaver Master连接 ...

  9. 监控和审计IBM InfoSphere BigInsights和Cloudera Hadoop的访问权限

    http://www.ithov.com/server/124456.shtml 您也将学习一个仅适用于 IBM InfoSphere BigInsights 的快速启动监控实现. 大数据骚动主要集中 ...

  10. Linux下的目录结构

    1./ -根 每个文件和目录从根目录开始 只有root用户具有该目录下的写权限,请注意,/root是root用户的主目录,这与/.不一样. 2. /bin  -用户二进制文件 包含二进制可执行的文件 ...