【BZOJ3196】二逼平衡树(树状数组,线段树)

题面

BZOJ题面

题解

如果不存在区间修改操作:

搞一个权值线段树

区间第K大--->直接在线段树上二分

某个数第几大--->查询一下区间的size和

某个数的前缀--->先查一下他是区间第几大,再求他-1大

某个数的后缀--->和上面那个有区别吗???

现在有了区间修改操作

多搞一个树状数组

套在一起就好啦

暴力开点开不下的

要动态开点

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<set>
#include<map>
#include<vector>
#include<queue>
using namespace std;
#define MAX 100000
inline int read()
{
int x=0,t=1;char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=-1,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return x*t;
}
struct Node
{
int ls,rs;
int v;
}t[MAX<<7];
int tot,S[MAX],rt[MAX],n,m,a[MAX];
int t1,t2,tmp1[MAX],tmp2[MAX];
int opt[MAX],ll[MAX],rr[MAX],kk[MAX];
int Tot;
inline int lowbit(int x){return x&(-x);}
void Modify(int &now,int l,int r,int pos,int w)
{
if(!now)now=++Tot;
t[now].v+=w;
if(l==r)return;
int mid=(l+r)>>1;
if(pos<=mid)Modify(t[now].ls,l,mid,pos,w);
else Modify(t[now].rs,mid+1,r,pos,w);
}
void PreModify(int pos,int w){for(int i=pos;i<=n;i+=lowbit(i))Modify(rt[i],1,tot,a[pos],w);}
int Query(int now,int l,int r,int pos)
{
if(l==r)return t[now].v;
int mid=(l+r)>>1;
if(pos<=mid)return Query(t[now].ls,l,mid,pos);
else return t[t[now].ls].v+Query(t[now].rs,mid+1,r,pos);
}
int Rank(int l,int r,int k)
{
if(l>r)return 0;
l-=1;int ret=0;
for(int i=r;i;i-=lowbit(i))ret+=Query(rt[i],1,tot,k);
for(int i=l;i;i-=lowbit(i))ret-=Query(rt[i],1,tot,k);
return ret;
}
int Kth(int l,int r,int k)
{
if(l==r)return l;
int sum=0,mid=(l+r)>>1;
for(int i=1;i<=t1;++i)sum+=t[t[tmp1[i]].ls].v;
for(int i=1;i<=t2;++i)sum-=t[t[tmp2[i]].ls].v;
if(sum>=k)
{
for(int i=1;i<=t1;++i)tmp1[i]=t[tmp1[i]].ls;
for(int i=1;i<=t2;++i)tmp2[i]=t[tmp2[i]].ls;
return Kth(l,mid,k);
}
else
{
for(int i=1;i<=t1;++i)tmp1[i]=t[tmp1[i]].rs;
for(int i=1;i<=t2;++i)tmp2[i]=t[tmp2[i]].rs;
return Kth(mid+1,r,k-sum);
}
}
int PreKth(int l,int r,int k)
{
l--;t1=t2=0;
for(int i=r;i;i-=lowbit(i))tmp1[++t1]=rt[i];
for(int i=l;i;i-=lowbit(i))tmp2[++t2]=rt[i];
return S[Kth(1,tot,k)];
}
int main()
{
n=read();m=read();
for(int i=1;i<=n;++i)a[i]=S[++tot]=read();
for(int i=1;i<=m;++i)
{
opt[i]=read();
if(opt[i]!=3)
{
ll[i]=read();rr[i]=read();kk[i]=read();
if(opt[i]!=2)S[++tot]=kk[i];
}
else
{
ll[i]=rr[i]=read();
S[++tot]=kk[i]=read();
}
}
sort(&S[1],&S[tot+1]);
tot=unique(&S[1],&S[tot+1])-S-1;
for(int i=1;i<=n;++i)a[i]=lower_bound(&S[1],&S[tot+1],a[i])-S,PreModify(i,1);
for(int i=1;i<=m;++i)
{
if(opt[i]==1)
{
kk[i]=lower_bound(&S[1],&S[tot+1],kk[i])-S;
printf("%d\n",Rank(ll[i],rr[i],kk[i]-1)+1);
}
else if(opt[i]==2)
{
printf("%d\n",PreKth(ll[i],rr[i],kk[i]));
}
else if(opt[i]==3)
{
PreModify(ll[i],-1);
a[ll[i]]=lower_bound(&S[1],&S[tot+1],kk[i])-S;
PreModify(ll[i],1);
}
else if(opt[i]==4)
{
kk[i]=lower_bound(&S[1],&S[tot+1],kk[i])-S;
int gg=Rank(ll[i],rr[i],kk[i]-1);
if(!gg)puts("-2147483647");
else printf("%d\n",PreKth(ll[i],rr[i],gg));
}
else
{
kk[i]=lower_bound(&S[1],&S[tot+1],kk[i])-S;
int gg=Rank(ll[i],rr[i],kk[i]);
if(gg==rr[i]-ll[i]+1)puts("2147483647");
else printf("%d\n",PreKth(ll[i],rr[i],gg+1));
}
}
return 0;
}

【BZOJ3196】二逼平衡树(树状数组,线段树)的更多相关文章

  1. 树状数组 && 线段树应用 -- 求逆序数

    参考:算法学习(二)——树状数组求逆序数 .线段树或树状数组求逆序数(附例题) 应用树状数组 || 线段树求逆序数是一种很巧妙的技巧,这个技巧的关键在于如何把原来单纯的求区间和操作转换为 求小于等于a ...

  2. 洛谷P2414 阿狸的打字机 [NOI2011] AC自动机+树状数组/线段树

    正解:AC自动机+树状数组/线段树 解题报告: 传送门! 这道题,首先想到暴力思路还是不难的,首先看到y有那么多个,菜鸡如我还不怎么会可持久化之类的,那就直接排个序什么的然后按顺序做就好,这样听说有7 ...

  3. hdu1394(枚举/树状数组/线段树单点更新&区间求和)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1394 题意:给出一个循环数组,求其逆序对最少为多少: 思路:对于逆序对: 交换两个相邻数,逆序数 +1 ...

  4. hdu 1166:敌兵布阵(树状数组 / 线段树,入门练习题)

    敌兵布阵 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submis ...

  5. hdu 5147 Sequence II【树状数组/线段树】

    Sequence IITime Limit: 5000/2500 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Problem ...

  6. BZOJ_1901_&_ZJU_2112_Dynamic_Rankings_(主席树+树状数组/线段树+(Treap/Splay))

    描述 http://www.lydsy.com/JudgeOnline/problem.php?id=1901 给出一个长度为n的数列A,有m次询问,询问分两种:1.修改某一位置的值;2.求区间[l, ...

  7. Color the ball(树状数组+线段树+二分)

    Color the ball Time Limit : 9000/3000ms (Java/Other)   Memory Limit : 32768/32768K (Java/Other) Tota ...

  8. 【bzoj4785】[Zjoi2017]树状数组 线段树套线段树

    题目描述 漆黑的晚上,九条可怜躺在床上辗转反侧.难以入眠的她想起了若干年前她的一次悲惨的OI 比赛经历.那是一道基础的树状数组题.给出一个长度为 n 的数组 A,初始值都为 0,接下来进行 m 次操作 ...

  9. hdu 1166 敌兵布阵——(区间和)树状数组/线段树

    pid=1166">here:http://acm.hdu.edu.cn/showproblem.php?pid=1166 Input 第一行一个整数T.表示有T组数据. 每组数据第一 ...

  10. 数据结构--树状数组&&线段树--基本操作

    随笔目的:方便以后对树状数组(BIT)以及基本线段树的回顾 例题链接:http://acm.hdu.edu.cn/showproblem.php?pid=1166 例题:hdu 1166 敌兵布阵 T ...

随机推荐

  1. centos7 网桥的配置

    centos7下配置网桥,两个步骤:1.新建网桥配置2.修改网卡配置 新建br0 网桥配置 在/etc/sysconfig/network-scripts/目录下新建ifcfg-br0,添加如下配置信 ...

  2. 一个网卡配置多个ip配置实现,centos7系统

    仅一个网卡情况下,配置多个ip可以让该设备通过几个ip被访问,或隐藏常用ip,让其他人访问临时ip 一.永久性增加一个IP 方法1: vim /etc/sysconfig/network-script ...

  3. [译]Serilog Tutorial

    在过去的几年中,结构化日志已经大受欢迎.而Serilog是 .NET 中最著名的结构化日志类库 ,我们提供了这份的精简指南来帮助你快速了解并运用它. 0. 内容 设定目标 认识Serilog 事件和级 ...

  4. MySQL Server 5.0安装教程

    相信很多朋友刚开始接触mysql数据库服务器,下面是mysql的安装教程,步骤明细也有详细的说明. 工具/原料   mysql MySQL安装的图解   1 打开下载的mysql安装文件mysql-5 ...

  5. Activiti中的各个service的作用

    各个Service的作用: RepositoryService 管理流程定义 RuntimeService 执行管理,包括启动.推进.删除流程实例等操作 TaskService 任务管理 Histor ...

  6. Asp.Net Core Identity+EFCore + Mysql踩坑记录

    搭建基础框架准备试试传说中的Identity,本以为很顺利,结果一路踩了N多坑 遂就把过程记录下来.方便自己以后查看,也希望能帮到遇到同样问题的朋友. 1.首先,引入Identity需要的类库,还有M ...

  7. 独立游戏大电影 原名(Indie.Game)

    电影链接 独立游戏大电影 感觉很不错呢!!

  8. AGC010 - A: Addition

    原题链接 题意简述 给出一个个数的数列,每次选出两个奇偶性相同的数合成一个数,问最终能否只剩下一个数. 分析 非常简单的一道题. 两个偶数可以合成一个偶数,两个奇数也能合成一个偶数.所以合并偶数时偶数 ...

  9. 统计表中 重复出现 XX次以上的数据

    在平时使用数据库查询数据时 经常会遇到查询表中出现XX次以上的数据  以前自己遇到就直接百度  然后拿来就用  在过段时间遇到就懵逼了 还得百度.... so  还是加深理解下  省的以后遇到再次一脸 ...

  10. iOS学习 NSString常用技巧

    字符串是程序设计最常用的数据类型之一了.在Mac/iPhone编程中,苹果为我们提供了一个不同的字符串类型NSString.有别与普通的String为数据类型,NSString其实是一个对象类型.NS ...