题面戳我

您需要写一种数据结构来维护一个有序数列,其中需要提供以下操作:

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

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

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

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

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

时空限制:2s,128M

\(n,m≤5⋅10^4\) 保证有序序列所有值在任何时刻满足\([0,108^]\)

sol

树状数组套动态开节点的权值线段树。

你需要完成以下这一系列操作:

\(Update(i,type)\),把原数组中\(i\)位置上的数进行\(type(=1 or-1)\)操作放到树状数组里面去。类比于树状数组的单点修改操作,这个的复杂度是\(O(\log_{2}^{2}n)\)

\(Sum(l,r,ql,qr)\),统计位置位于\([l,r]\),且权值位于\([ql,qr]\)之间的树的个数。易知我们可以拿\(\log n\)棵线段树跟另外\(\log n\)棵线段树作差。复杂度\(O(\log_{2}^{2}n)\)

\(Rank(l,r,k)\),查找区间\([l,r]\)中第\(k\)大的数是多少。这个就是。。。那个呀。复杂度\(O(\log_{2}^{2}n)\)

我们来看这道题。首先数据离线然后离散化不解释。

操作一,排名就是小于它的数的个数+1。直接调用\(Sum(l,r,1,k-1)\)

操作二,直接调用\(Rank(l,r,k)\)

操作三,两次调用\(Update(pos,type)\),第一次\(type=-1\),在原数组里改掉以后调用\(type=1\)

操作四,前驱。先查\(Sum(l,r,1,k-1)\),若为0则输出-2147483647,否则输出\(Rank(l,r,sum)\)(sum就是前面查的答案)

操作四,后继。同理。先查\(Sum(l,r,1,k)\),若为r-l+1则输出2147483647,否则输出\(Rank(l,r,sum+1)\)

操作四五比较难搞,这里要敲黑板。

然后这题就讲完了。

code

#include<cstdio>
#include<algorithm>
using namespace std;
const int N = 50005;
struct tree{int ls,rs,num;}t[N*200];
struct query{int opt,l,r,pos,k;}q[N];
int n,m,a[N],o[N<<1],len,rt[N],tot,temp[2][20],cnt[2];
int gi()
{
int x=0,w=1;char ch=getchar();
while ((ch<'0'||ch>'9')&&ch!='-') ch=getchar();
if (ch=='-') w=0,ch=getchar();
while (ch>='0'&&ch<='9') x=(x<<3)+(x<<1)+ch-'0',ch=getchar();
return w?x:-x;
}
void Modify(int &x,int l,int r,int pos,int v)
{
if (!x) x=++tot;
t[x].num+=v;
if (l==r) return;
int mid=l+r>>1;
if (pos<=mid) Modify(t[x].ls,l,mid,pos,v);else Modify(t[x].rs,mid+1,r,pos,v);
}
int Query(int x,int l,int r,int ql,int qr)
{
if (l>=ql&&r<=qr) return t[x].num;
int mid=l+r>>1,s=0;
if (ql<=mid) s+=Query(t[x].ls,l,mid,ql,qr);
if (qr>mid) s+=Query(t[x].rs,mid+1,r,ql,qr);
return s;
}
int Find(int l,int r,int k)
{
if (l==r) return l;
int mid=l+r>>1,sum=0;
for (int i=1;i<=cnt[1];i++) sum+=t[t[temp[1][i]].ls].num;
for (int i=1;i<=cnt[0];i++) sum-=t[t[temp[0][i]].ls].num;
if (k<=sum)
{
for (int i=1;i<=cnt[1];i++) temp[1][i]=t[temp[1][i]].ls;
for (int i=1;i<=cnt[0];i++) temp[0][i]=t[temp[0][i]].ls;
return Find(l,mid,k);
}
else
{
for (int i=1;i<=cnt[1];i++) temp[1][i]=t[temp[1][i]].rs;
for (int i=1;i<=cnt[0];i++) temp[0][i]=t[temp[0][i]].rs;
return Find(mid+1,r,k-sum);
}
}
void Update(int pos,int v)
{
for (int i=pos;i<=n;i+=i&-i)
Modify(rt[i],1,len,a[pos],v);
}
int Sum(int l,int r,int ql,int qr)
{
if (ql>qr) return 0;
int sum=0;
for (int j=r;j;j-=j&-j) sum+=Query(rt[j],1,len,ql,qr);
for (int j=l-1;j;j-=j&-j) sum-=Query(rt[j],1,len,ql,qr);
return sum;
}
int Rank(int l,int r,int k)
{
cnt[1]=cnt[0]=0;
for (int j=r;j;j-=j&-j) temp[1][++cnt[1]]=rt[j];
for (int j=l-1;j;j-=j&-j) temp[0][++cnt[0]]=rt[j];
return o[Find(1,len,k)];
}
int main()
{
n=gi();m=gi();
for (int i=1;i<=n;i++) o[++len]=a[i]=gi();
for (int i=1;i<=m;i++)
{
q[i].opt=gi();
if (q[i].opt!=3) q[i].l=gi(),q[i].r=gi();
else q[i].pos=gi();
q[i].k=gi();
if (q[i].opt!=2) o[++len]=q[i].k;
}
sort(o+1,o+len+1);
len=unique(o+1,o+len+1)-o-1;
for (int i=1;i<=n;i++)
{
a[i]=lower_bound(o+1,o+len+1,a[i])-o;
Update(i,1);
}
for (int i=1;i<=m;i++)
{
if (q[i].opt!=2) q[i].k=lower_bound(o+1,o+len+1,q[i].k)-o;
if (q[i].opt==1)
printf("%d\n",Sum(q[i].l,q[i].r,1,q[i].k-1)+1);
if (q[i].opt==2)
printf("%d\n",Rank(q[i].l,q[i].r,q[i].k));
if (q[i].opt==3)
{
Update(q[i].pos,-1);
a[q[i].pos]=q[i].k;
Update(q[i].pos,1);
}
if (q[i].opt==4)
{
int sum=Sum(q[i].l,q[i].r,1,q[i].k-1);
if (!sum) printf("-2147483647\n");
else printf("%d\n",Rank(q[i].l,q[i].r,sum));
}
if (q[i].opt==5)
{
int sum=Sum(q[i].l,q[i].r,1,q[i].k);
if (sum==q[i].r-q[i].l+1) printf("2147483647\n");
else printf("%d\n",Rank(q[i].l,q[i].r,sum+1));
} }
return 0;
}

[Lugu3380]【模板】二逼平衡树(树套树)的更多相关文章

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

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

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

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

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

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

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

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

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

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

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

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

  9. P3380 【模板】二逼平衡树(树套树)(线段树套平衡树)

    P3380 [模板]二逼平衡树(树套树) 前置芝士 P3369 [模板]普通平衡树 线段树套平衡树 这里写的是线段树+splay(不吸氧竟然卡过了) 对线段树的每个节点都维护一颗平衡树 每次把给定区间 ...

  10. 洛谷 P3380 bzoj3196 Tyvj1730 【模板】二逼平衡树(树套树)

    [模板]二逼平衡树(树套树) 题目描述 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作: 查询k在区间内的排名 查询区间内排名为k的值 修改某一位值上的数值 查询k在 ...

随机推荐

  1. Eclipse Java,debug模式无法调试,调试按钮不可用时解决办法

    经常出现debug模式进入后,debug的几个按钮置灰,F5,6,8,没有任何反应时,这样操作: 退出 Eclipse.打开 Eclipse 目录下的 configuration 下的 org.ecl ...

  2. shell脚本实现nfs服务安装配置,共享文件分发

    ##############################Deploy nfs######################## echo "start deploy nfs-server& ...

  3. angular2^ typescript 将 文件和Json数据 合并发送到服务器(1.客户端处理)

    首先介绍下框架基本流程   (web > webservice  [前端架构] ) > (nodejs [ 数据中转站 ]) >(api [后台接口]) --web (html  a ...

  4. ABP框架源码学习之授权逻辑

    asp.net core的默认的几种授权方法参考"雨夜朦胧"的系列博客,这里要强调的是asp.net core mvc中的授权和asp.net mvc中的授权不一样,建议先看前面& ...

  5. Java经典编程题50道之二十五

    一个5位数,判断它是不是回文数.即12321是回文数,个位与万位相同,十位与千位相同. public class Example25 {    public static void main(Stri ...

  6. w !sudo tee %

    w !sudo tee % 该命令可用于保存有权限的写文件

  7. mysql3 - 常规数据检索、常见操作与函数

    一.常规数据检索 二.常见操作与函数

  8. 浅谈session,cookie,sessionStorage,localStorage的区别及应用场景

    浏览器的缓存机制提供了可以将用户数据存储在客户端上的方式,可以利用cookie,session等跟服务端进行数据交互. 一.cookie和session cookie和session都是用来跟踪浏览器 ...

  9. hihoCoder 1044 : 状态压缩·一 状压dp

    思路:状态压缩,dp(i, j)表示考虑前i个数且[i-m+1, i]的选择情况为j.如果要选择当前这个数并且,数位1的个数不超过q,则dp[i+1][nex] = max(dp[i+1][nex], ...

  10. CodeForces-740B Alyona and flowers

    题目要求选择一些花的集合,如果暴力去枚举每种选择方法明显是不行的.换种方式考虑:每一个集合都能为最后的答案做出要么正的.要么负的.要么0贡献,先把所有集合能做出的贡献预处理,然后从m个集合里面选择贡献 ...