BZOJ.3489.A simple rmq problem(主席树 Heap)
当时没用markdown写,可能看起来比较难受...可以复制到别的地方看比如DevC++。
\(Description\)
给定一个长为n的序列,多次询问[l,r]中最大的只出现一次的数。强制在线。
\(Solution\)
我也不知道该怎么说,反正就是预处理 建主席树,套堆/set,树i存l为i,r为[i,n]的答案(这样就是在某棵树上单点查maxv了)。
处理好最初的树,就可以利用主席树根据前缀建树的性质,每个点i的建树只需要处理i位置。
那么最初的树就是将所有的数在区间[第一次出现位置,nxt[]-1]中加入这个数。
当左端点i移动时,A[i-1]的贡献没了,要删掉;但是如果A[i-1]在后面出现,则还要在[nxt[A[i-1]],nxt[nxt[A[i-1]]]-1]上加入A[i-1]。(A[i]已经处理了,要么在建最初树时要么在先前建的树中)
也不是每棵树每个节点都套堆,堆只需要帮助建主席树时加加删删(还是利用前缀),要用的只是根据堆得到的每个节点的maxv[],所以堆还是4n大就可以。。
和CF那题不同的是这题固定l更方便,那题固定r好些。所以只需要nxt(和最靠前的一个las用来建最初的树)。
第一次主席树区间修改。。在信息覆盖当前区间时直接给节点赋值,return;查询单点时加上沿路节点的mx[](标记永久化)。
注意新建节点时copy x的mx[]。因为每次删除是完全删除上一次的加入,so在未递归到完全覆盖区间前copy mx是对的?真这样麻烦吗。。以后是不是该乖乖写Update()。。
询问的限制条件可以看成三个维度,即在某三维空间内找权值最大的点,可以用K-D Tree。出题人说这不应是正解,可以卡,只是数据问题跑的比主席树快。。可以看这儿。
不过还是K-D Tree更好的应用吧,不想看了,先放着。。
K-D Tree:https://blog.csdn.net/FromATP/article/details/61198101
代码参考(chao)自:https://blog.csdn.net/u014609452/article/details/51396271。
非强制在线可以直接线段树+排序扫描线做,有个简化版的题见这儿。
//147544kb 9488ms
#include <queue>
#include <cstdio>
#include <cctype>
#include <algorithm>
//#define gc() getchar()
#define MAXIN 300000
#define gc() (SS==TT&&(TT=(SS=IN)+fread(IN,1,MAXIN,stdin),SS==TT)?EOF:*SS++)
const int N=1e5+5;
int n,Q,A[N],las[N],nxt[N],rcnt,root[N*3],pos[N];
char IN[MAXIN],*SS=IN,*TT=IN;
struct Tree//Persistent Segment Tree
{
#define S N*100//我也不知道这样搞区间修改的主席树会有多少节点(2个log?)
#define lson son[x][0]
#define rson son[x][1]
#define ToL l,m,rt<<1
#define ToR m+1,r,rt<<1|1
struct Heap
{
std::priority_queue<int> h,d;
inline void Insert(int x) {h.push(x);}
inline void Delete(int x) {d.push(x);}
inline void Maintain(){
while(!h.empty()&&!d.empty()&&h.top()==d.top()) h.pop(),d.pop();
}
inline int Top(){
Maintain(); return h.empty()?0:h.top();
}
}hp[N<<2];
int tot,son[S][2],mx[S];
// inline void Update(int x){//Update没啥用啊 又不用区间信息
// mx[x]=std::max(mx[lson],mx[rson]);
// }
void Insert(int &y,int x,int l,int r,int rt,int L,int R,int v)//rt:一般线段树的节点 存储堆
{
y=++tot;
if(L<=l && r<=R){
hp[rt].Insert(v), mx[y]=hp[rt].Top(), son[y][0]=lson, son[y][1]=rson;
return ;
}
int m=l+r>>1; mx[y]=mx[x];//!
if(L<=m)
if(m<R) Insert(son[y][0],lson,ToL,L,R,v), Insert(son[y][1],rson,ToR,L,R,v);
else son[y][1]=rson, Insert(son[y][0],lson,ToL,L,R,v);
else son[y][0]=lson, Insert(son[y][1],rson,ToR,L,R,v);
}
void Delete(int &y,int x,int l,int r,int rt,int L,int R,int v)
{
y=++tot;
if(L<=l && r<=R){
hp[rt].Delete(v), mx[y]=hp[rt].Top(), son[y][0]=lson, son[y][1]=rson;
return;
}
int m=l+r>>1; mx[y]=mx[x];//!
if(L<=m)
if(m<R) Delete(son[y][0],lson,ToL,L,R,v), Delete(son[y][1],rson,ToR,L,R,v);
else son[y][1]=rson, Delete(son[y][0],lson,ToL,L,R,v);
else son[y][0]=lson, Delete(son[y][1],rson,ToR,L,R,v);
}
int Query(int x,int l,int r,int pos)
{
if(!x) return 0;//这个没啥用啊...
if(l==r) return mx[x];
int m=l+r>>1;
if(pos<=m) return std::max(mx[x],Query(lson,l,m,pos));
else return std::max(mx[x],Query(rson,m+1,r,pos));
}
}T;
inline int read()
{
int now=0;register char c=gc();
for(;!isdigit(c);c=gc());
for(;isdigit(c);now=now*10+c-'0',c=gc());
return now;
}
int main()
{
n=read(), Q=read();
for(int i=1; i<=n; ++i) A[i]=read();
for(int i=1; i<=n; ++i) las[i]=n+1;
for(int i=n; i; --i) nxt[i]=las[A[i]], las[A[i]]=i;
for(int i=n; i; --i)//直接枚举值域建树就可以啊 话说倒着插入堆会更快吗
if(las[i]<=n) T.Insert(root[rcnt],root[rcnt++],1,n,1,las[i],nxt[las[i]]-1,i);//参数顺序
pos[1]=root[rcnt];//root[rcnt] not rcnt→_→
for(int i=1; i<n; ++i)
{
T.Delete(root[rcnt],root[rcnt++],1,n,1,i,nxt[i]-1,A[i]);
if(nxt[i]<=n)
T.Insert(root[rcnt],root[rcnt++],1,n,1,nxt[i],nxt[nxt[i]]-1,A[i]);
pos[i+1]=root[rcnt];
}
for(int ans=0,i=1,l,r; i<=Q; ++i)
{
l=(read()+ans)%n+1, r=(read()+ans)%n+1;
if(l>r) std::swap(l,r);//l>r&&(std::swap(l,r),1);
printf("%d\n",ans=T.Query(pos[l],1,n,r));
}
return 0;
}
BZOJ.3489.A simple rmq problem(主席树 Heap)的更多相关文章
- bzoj 3489 A simple rmq problem——主席树套线段树
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3489 题解:http://www.itdaan.com/blog/2017/11/24/9b ...
- bzoj 3489 A simple rmq problem —— 主席树套线段树
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3489 题解:http://www.itdaan.com/blog/2017/11/24/9b ...
- bzoj 3489: A simple rmq problem k-d树思想大暴力
3489: A simple rmq problem Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 551 Solved: 170[Submit][ ...
- bzoj 3489 A simple rmq problem - 线段树
Description 因为是OJ上的题,就简单点好了.给出一个长度为n的序列,给出M个询问:在[l,r]之间找到一个在这个区间里只出现过一次的数,并且要求找的这个数尽可能大.如果找不到这样的数,则直 ...
- BZOJ 3489: A simple rmq problem
3489: A simple rmq problem Time Limit: 40 Sec Memory Limit: 600 MBSubmit: 1594 Solved: 520[Submit] ...
- [BZOJ 3489] A simple rmq problem 【可持久化树套树】
题目链接:BZOJ - 3489 题目分析 “因为是OJ上的题,就简单点好了.”——出题人 真的..好..简单... 首先,我们求出每个数的前一个与它相同的数的位置,即 prev[i] ,如果前面没有 ...
- BZOJ 3489 A simple rmq problem 可持久化KDtree/二维线段树
题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=3489 题意概述: 给出一个序列,每次询问一个序列区间中仅出现了一次的数字最大是多少,如果 ...
- BZOJ 3489 A simple rmq problem(可持久化线段树)
题目链接:http://www.lydsy.com:808/JudgeOnline/problem.php?id=3489 题意:一个数列.每次询问一个区间内出现一次的最大的数字是多少. 思路:设la ...
- bzoj 3585: mex && 3339: Rmq Problem -- 主席树
3585: mex Time Limit: 20 Sec Memory Limit: 128 MB Description 有一个长度为n的数组{a1,a2,...,an}.m次询问,每次询问一个区 ...
随机推荐
- Spring4笔记6--Spring与AOP
Spring与AOP: AOP的引入: 主业务经常需要调用系统级业务(交叉业务),如果在主业务代码中大量的调用系统级业务代码,会使系统级业务与主业务深度耦合在一起,大大影响了主业务逻辑的可读性,降低了 ...
- 【FCS NOI2018】福建省冬摸鱼笔记 day6【FJOI 2018】福建省选混分滚蛋记 day1
记录一下day6发生的事情吧. 7:30 到达附中求索碑,被人膜,掉RP. 7:50 进考场,6楼的最后一排的最左边的位置,世界上最角落的地方,没有任何想法. 发现电脑时间和别人不一样,赶快调了一下. ...
- Memcached服务器UDP反射放大攻击
1.前言 2月28日,Memcache服务器被曝出存在UDP反射放大攻击漏洞.攻击者可利用这个漏洞来发起大规模的DDoS攻击,从而影响网络正常运行.漏洞的形成原因为Memcache 服务器UDP 协议 ...
- libevent简介和使用【转】
转自:http://www.open-open.com/lib/view/open1386510630330.html libevent是一个基于事件触发的网络库,memcached底层也是使用lib ...
- Hyperledger Fabric1.0.0搭建
系统环境: 阿里云新装的Centos7.2 yum -y update yum install -y openssl openssl-devel gcc gcc-c++ zlib zlib-devel ...
- Oracle基础结构认知—初识oracle【转】
Oracle服务器(oracle server)由实例和数据库组成.其中,实例就是所谓的关系型数据库管理系统(Relational Database Management System,RDBMS), ...
- linux系统时钟和硬件时钟不一致
在做DB2 集群复制的时候要求两台主机时间相互一致. 但是在一台主机上系统时间和硬件时间相差12个小时左右:手动同步后,重启后又相差12个小时左右. 为什么会是这样的,先介绍下系统时钟和硬件时钟的区别 ...
- Session和Cookie,Django的自动登录机制
什么是Cookie? Cookie是浏览器的本地存储机制,存储服务器返回的各种信息,下次发起请求时再发送给服务端,比如访问baidu 什么是Session? 刚才说道,Cookie存储服务端返回的信息 ...
- NVME SSD vs SATA SSD(转)
NVMe是个啥?未来SSD主流标准早知 关注固态硬盘的朋友应该对于这个词汇并不陌生,特别是今年NVMe也频繁出现在各大媒体文章中,随着高端SSD市场逐渐从SATA专项PCI-E时,以前的AHCI标准已 ...
- css-概述和选择器
一:CSS 概述 CSS 指层叠样式表 (Cascading Style Sheets) 样式定义如何显示 HTML 元素 样式通常存储在样式表中 把样式添加到 HTML 4.0 中,是为了解决内容与 ...