Luogu P2617 Dynamic Rankings(整体二分)
题目
动态区间第K小模板题。
一个非常可行的办法是BIT套动态开点权值SegTree,但是它跑的实在太慢了。
然后由于这题并没有强制在线,所以我们可以使用整体二分来吊打树套树。
当然如果强制在线的话就只能够用大常数的树套树了。
这里稍微结合这道题讲一下整体二分吧。
我们把所有询问加修改(统称为操作)按时间顺序排好,然后一起二分答案。
对于一个操作区间\([ql,qr]\)和答案\(mid\),我们顺序遍历所有操作。
将所有能在当前情况下产生影响的修改进行执行并放到左操作区间,否则直接放到右操作区间。
将所有答案\(\le mid\)的询问放到左操作区间,否则在处理完当前执行的操作的影响之后放到右区间。
然后把我们用来维护修改求答案的数据结构还原。并对左操作区间和右操作区间分别分治。
因为我们考虑这样一个事实:不管我们分治到了什么时候,所有操作区间中的操作还是按时间排序的。
因此我们可以直接顺序遍历操作来修改和查询。
而在考虑左操作区间的修改对右操作区间的查询的影响时,我们可以直接在遍历的过程中进行修改。
在这个题目中,我们二分出来一个答案\(mid\),那么我们把所有\(\le mid\)的数视作\(1\),否则视作\(0\),那么我们就可以根据前缀和来判断答案是\(\le mid\)还是\(>mid\)了。
我们可以将初始序列看做在某个位置插入一个数,修改看做在某个位置把原来的数删掉再插入一个新的。
那么我们就可以使用BIT来支持单点修改和区间查询。
注意在清零BIT的时候是要撤销而非memset,否则复杂度原地升天。
#include<bits/stdc++.h>
using namespace std;
namespace IO
{
char ibuf[(1<<21)+1],obuf[(1<<21)+1],st[15],*iS,*iT,*oS=obuf,*oT=obuf+(1<<21);
char Get(){return (iS==iT? (iT=(iS=ibuf)+fread(ibuf,1,(1<<21)+1,stdin),(iS==iT? EOF:*iS++)):*iS++);}
void Flush(){fwrite(obuf,1,oS-obuf,stdout),oS=obuf;}
void Put(char x){*oS++=x;if(oS==oT)Flush();}
char fetch(){char c=Get();while(!isupper(c))c=Get();return c;}
int read(){int x=0,c=Get();while(!isdigit(c))c=Get();while(isdigit(c))x=x*10+c-48,c=Get();return x;}
void write(int x){int top=0;if(!x)Put('0');while(x)st[++top]=(x%10)+48,x/=10;while(top)Put(st[top--]);Put('\n');}
}
using namespace IO;
const int N=100007;
struct node{int opt,l,r,x,id;}q[N*3],q1[N*3],q2[N*3];
int n,m,Q,tot,a[N],h[N<<1],t[N],ans[N*3];
void add(int p,int v){for(;p<=n;p+=p&-p)t[p]+=v;}
int ask(int p){int s=0;for(;p;p^=p&-p)s+=t[p];return s;}
void solve(int ql,int qr,int vl,int vr)
{
if(ql>qr) return ;
if(vl==vr)
{
for(int i=ql;i<=qr;++i) if(q[i].opt==2) ans[q[i].id]=h[vl];
return ;
}
int mid=(vl+vr)>>1,l=0,r=0;
for(int i=ql,k;i<=qr;++i)
if(q[i].opt==1)
if(q[i].r<=mid) add(q[i].l,q[i].x),q1[l++]=q[i];
else q2[r++]=q[i];
else
if((k=ask(q[i].r)-ask(q[i].l-1))>=q[i].x) q1[l++]=q[i];
else q[i].x-=k,q2[r++]=q[i];
for(int i=0;i<l;++i) if(q1[i].opt==1) add(q1[i].l,-q1[i].x);
memcpy(q+ql,q1,l*(sizeof(node))),memcpy(q+ql+l,q2,r*(sizeof(node)));
solve(ql,ql+l-1,vl,mid),solve(ql+l,qr,mid+1,vr);
}
int main()
{
n=read(),m=read();
for(int i=1;i<=n;++i) h[++tot]=a[i]=read(),q[++Q]=(node){1,i,a[i],1,Q};
for(int i=1,x,l,r;i<=m;++i)
if(fetch()=='Q') l=read(),r=read(),x=read(),q[++Q]=(node){2,l,r,x,Q};
else l=read(),h[++tot]=r=read(),q[++Q]=(node){1,l,a[l],-1,Q},q[++Q]=(node){1,l,a[l]=r,1,Q};
sort(h+1,h+tot+1),tot=unique(h+1,h+tot+1)-h-1;
for(int i=1;i<=n;++i) a[i]=lower_bound(h+1,h+tot+1,a[i])-h;
for(int i=1;i<=Q;++i) if(q[i].opt==1) q[i].r=lower_bound(h+1,h+tot+1,q[i].r)-h;
solve(1,Q,1,tot);
for(int i=1;i<=Q;++i) if(ans[i]) write(ans[i]);
return Flush(),0;
}
Luogu P2617 Dynamic Rankings(整体二分)的更多相关文章
- 洛谷$P2617\ Dynamic\ Rankings$ 整体二分
正解:整体二分 解题报告: 传送门$w$ 阿查询带修区间第$k$小不显然整体二分板子呗,,, 就考虑先按时间戳排序(,,,其实并不需要读入的时候就按着时间戳排的鸭$QwQ$ 每次二分出$mid$先把所 ...
- [bzoj1901][zoj2112][Dynamic Rankings] (整体二分+树状数组 or 动态开点线段树 or 主席树)
Dynamic Rankings Time Limit: 10 Seconds Memory Limit: 32768 KB The Company Dynamic Rankings has ...
- 【BZOJ1901】Dynamic Rankings [整体二分]
Dynamic Rankings Time Limit: 10 Sec Memory Limit: 128 MB[Submit][Status][Discuss] Description 给定一个含 ...
- BZOJ 1901 Zju2112 Dynamic Rankings ——整体二分
[题目分析] 上次用树状数组套主席树做的,这次用整体二分去水. 把所有的查询的结果一起进行二分,思路很好. [代码] #include <cstdio> #include <cstr ...
- BZOJ1901: Zju2112 Dynamic Rankings(整体二分 树状数组)
Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 9094 Solved: 3808[Submit][Status][Discuss] Descript ...
- ZOJ2112 Dynamic Rankings(整体二分)
今天学习了一个奇技淫巧--整体二分.关于整体二分的一些理论性的东西,可以参见XRH的<浅谈数据结构题的几个非经典解法>.然后下面是一些个人的心得体会吧,写下来希望加深一下自己的理解,或者如 ...
- BZOJ 1901 Dynamic Rankings (整体二分+树状数组)
题目大意:略 洛谷传送门 这道题在洛谷上数据比较强 貌似这个题比较常见的写法是树状数组套主席树,动态修改 我写的是整体二分 一开始的序列全都视为插入 对于修改操作,把它拆分成插入和删除两个操作 像$C ...
- luogu P2617 Dynamic Rankings && bzoj 1901 (带修改区间第k大)
链接:https://www.luogu.org/problemnew/show/P2617 思路: 如果直接在主席树上修改的话,每次修改都会对后面所有的树造成影响,一次修改的复杂度就会变成 : n* ...
- Luogu P2617 Dynamic Rankings
带修主席树的模板,因为状态不好所以敲了很长时间,不过写完感觉能更好地理解主席树了. 核心其实就是树状数组套主席树,维护方法不再是以前的那种一步一修改,而是对于树状数组上的每一个点建立一棵权值线段树,然 ...
随机推荐
- SpringMVC——入门
一.SpringMVC介绍: Spring Web MVC是一种基于Java的实现了Web MVC设计模式的请求驱动类型的轻量级Web框架,即使用了MVC架构模式的思想,将web层进行职责解耦,基于请 ...
- Vue_(组件)自定义指令
Vue.js自定义指令 传送门 自定义指令:除了内置指令,Vue也允许用户自定义指令 注册指令:通过全局API Vue.directive可以注册自定义指令 自定义指令的钩子函数参数:自定义指令的钩子 ...
- 深入聚焦 call,apply 和 bind
在JavaScript 中,call.apply 和 bind 是 Function 对象自带的三个方法,这三个方法的主要作用是改变函数中的 this 指向,从而可以达到`接花移木`的效果.本文将对这 ...
- 让SpringBoot工程支持热部署
下载地址:https://files.cnblogs.com/files/xiandedanteng/SpringBootWeb-1_20190928.rar 修改Java文件后,每次要重启才好用,修 ...
- TcpSendRcv方法笔记1
if (ns.DataAvailable) return ns.ReadByte(); ns.DataAvailable:获取一个值,该值指示在要读取的 NetworkStream 上是否有可用的数据 ...
- C#连接内存数据库redis【1、Redis存读取数据】
这一节演示下载.NET中怎样使用Redis存储数据.在.net中比较常用的客户端类库是ServiceStack,看下通过servicestack怎样存储数据. DLL下载:https://github ...
- Tomcat 8.5 配置 SSL 证书
前文: 1.以上内容仅支持Linux-Tomcat配置 正文: 说一下我遇到的坑,我使用的服务器是阿里云服务器,阿里云提供的云服务器Tomcat配置SSL是7.0版本,跟8.5出入较大. 以下为阿里提 ...
- 性能测试的 Check List (不断更新中)
1. 开发人员是否提交了测试申请?2. 测试对象是否已经明确?3. 测试范围是否已经明确?4. 本次不被测试的范围是否已经明确?5. 测试目标是否已经明确?6. 何时开始性能测试?7. 何时终止一轮性 ...
- 面向对象编程(oop)的变迁
作者:匿名用户链接:https://www.zhihu.com/question/34018003/answer/132740170来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请 ...
- Linux中如何批量删除目录下文件后缀
1. rename rename分为perl版本和C版本,以下截图是C版本效果: perl版本:rename 's/.bak//' *.bak 2. for循环+awk 3. for循环+cut 4. ...