题目链接


题意:多次求区间\(mex\)。

考虑\([1,i]\)的\(mex[i]\),显然是单调的

而对于\([l,r]\)与\([l+1,r]\),如果\(nxt[a[l]]>r\),那么\([l+1,r]\)中所有\(>a[l]\)的数显然要改成\(a[l]\)

把询问按左端点排序;离散化,预处理下\(nxt[]\),处理出\([1,i]\)的\(mex[i]\)。剩下就是线段树的区间更新、单点查询了

/*
离散化的时候>=n的全部看做n就好了
查询时是只需查r点的(l之前能更新r的已经更新完了,初始时是[1,r],r点现在就是[l,r]了)
单点即可不需要PushUp(也不好得某个区间的mex)
非叶节点上的mex完全可以代替tag
离散化需要注意 其实不是很懂这
*/
#include<cstdio>
#include<cctype>
#include<algorithm>
#define gc() getchar()
#define now node[rt]
#define lson node[node[rt].ls]
#define rson node[node[rt].rs]
const int N=2e5+5,INF=1e7; int n,m,A[N],mex[N]/*不要和A混用*/,tmp[N],nxt[N],las[N],ans[N];
bool vis[N];
struct Ques
{
int l,r,id;
Ques() {}
Ques(int l,int r,int id): l(l),r(r),id(id) {}
bool operator <(const Ques &x)const {return l<x.l;}
}q[N];
struct Seg_Tree
{
int tot;
struct Node
{
int ls,rs,mex;
}node[N<<1];
inline void Upd(int &x,int v) {x=std::min(x,v);}
inline void PushDown(int rt)
{
Upd(lson.mex,now.mex), Upd(rson.mex,now.mex);
now.mex=INF;
}
void Build(int l,int r)
{
int rt=tot++;
if(l==r) now.mex = mex[l];
else
{
int m=l+r>>1; now.mex=INF;
now.ls=tot, Build(l,m);
now.rs=tot, Build(m+1,r);
}
}
void Update(int l,int r,int rt,int L,int R,int v)
{
if(L<=l && r<=R) Upd(now.mex,v);
else
{
if(now.mex<INF) PushDown(rt);
int m=l+r>>1;
if(L<=m) Update(l,m,now.ls,L,R,v);
if(m<R) Update(m+1,r,now.rs,L,R,v);
}
}
int Query(int l,int r,int rt,int p)
{
if(l==r) return now.mex;
if(now.mex<INF) PushDown(rt);
int m=l+r>>1;
if(p<=m) return Query(l,m,now.ls,p);
else return Query(m+1,r,now.rs,p);
}
}t;
#undef now
inline int read()
{
int now=0,f=1;register char c=gc();
for(;!isdigit(c);c=gc()) if(c=='-') f=-1;
for(;isdigit(c);now=now*10+c-'0',c=gc());
return now*f;
}
int Find(int x,int r)
{
int l=1,m;
while(l<r)
{
if(tmp[(m=l+r>>1)]<x) l=m+1;
else r=m;
}
return l;
} int main()
{
n=read(),m=read();
for(int i=1; i<=n; ++i) tmp[i]=A[i]=std::min(n,read());
std::sort(tmp+1,tmp+1+n);
int cnt=1;
for(int i=2; i<=n && !(tmp[i]==n&&tmp[i+1]==n); ++i)
if(tmp[i]!=tmp[i-1]) tmp[++cnt]=tmp[i];
for(int k=0,p,i=1; i<=n; ++i)
{
vis[p=Find(A[i],cnt)]=1;
if(A[i]==k)//只有在当前最小值出现时才更新。。mex...
while(vis[p])//p-1,vis[k]?
{
++k;
if(tmp[++p]!=k) break;//离散化后
}
mex[i]=k;
}
t.Build(1,n);
for(int i=0; i<=n; ++i) las[i]=n+1;
for(int tp,i=n; i; --i) nxt[i]=las[tp=Find(A[i],cnt)], las[tp]=i;//!
for(int l,i=1; i<=m; ++i) l=read(), q[i]=Ques(l,read(),i);
std::sort(q+1,q+1+m);
for(int now=1,i=1; i<=m; ++i)
{
while(now<q[i].l)
t.Update(1,n,0,now+1,nxt[now]-1,A[now]), ++now;
ans[q[i].id]=t.Query(1,n,0,q[i].r);
}
for(int i=1; i<=m; ++i) printf("%d\n",ans[i]); return 0;
}

BZOJ.3585.mex(线段树)的更多相关文章

  1. bzoj 3585 mex - 线段树 - 分块 - 莫队算法

    Description 有一个长度为n的数组{a1,a2,...,an}.m次询问,每次询问一个区间内最小没有出现过的自然数. Input 第一行n,m. 第二行为n个数. 从第三行开始,每行一个询问 ...

  2. BZOJ 3585: mex [主席树]

    3585: mex Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 787  Solved: 422[Submit][Status][Discuss] ...

  3. BZOJ.4184.shallot(线段树分治 线性基)

    BZOJ 裸的线段树分治+线性基,就是跑的巨慢_(:з」∠)_ . 不知道他们都写的什么=-= //41652kb 11920ms #include <map> #include < ...

  4. [BZOJ 4025]二分图(线段树分治+带边权并查集)

    [BZOJ 4025]二分图(线段树分治+带边权并查集) 题面 给出一个n个点m条边的图,每条边会在时间s到t出现,问每个时间的图是否为一个二分图 \(n,m,\max(t_i) \leq 10^5\ ...

  5. 主席树||可持久化线段树+离散化 || 莫队+分块 ||BZOJ 3585: mex || Luogu P4137 Rmq Problem / mex

    题面:Rmq Problem / mex 题解: 先离散化,然后插一堆空白,大体就是如果(对于以a.data<b.data排序后的A)A[i-1].data+1!=A[i].data,则插一个空 ...

  6. BZOJ 3585: mex( 离线 + 线段树 )

    离线, 询问排序. 先处理出1~i的答案, 这样可以回答左端点为1的询问.完成后就用seq(1)将1到它下一次出现的位置前更新. 不断这样转移就OK了 ------------------------ ...

  7. bzoj 3585: mex && 3339: Rmq Problem -- 主席树

    3585: mex Time Limit: 20 Sec  Memory Limit: 128 MB Description 有一个长度为n的数组{a1,a2,...,an}.m次询问,每次询问一个区 ...

  8. HDU-4747 Mex 线段树

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4747 题意:求一个数列中,所有mex(L,R)的和. 注意到mex是单调不降的,那么首先预处理出mex ...

  9. 【BZOJ 3476】 线段树===

    59  懒惰的奶牛贝西所在的牧场,散落着 N 堆牧草,其中第 i 堆牧草在 ( Xi,Yi ) 的位置,数量有 Ai 个单位.贝西从家移动到某一堆牧草的时候,只能沿坐标轴朝正北.正东.正西.正南这四个 ...

随机推荐

  1. How to Repair GRUB2 When Ubuntu Won’t Boot

    Ubuntu and many other Linux distributions use the GRUB2 boot loader. If GRUB2 breaks—for example, if ...

  2. 使用Jyhon脚本和PMI模块监控WAS性能数据

    使用Jyhon脚本和PMI模块监控WAS性能数据的优点有: 1.可以使用非交互的方式远程获取数据 2.不需要图形化模块支持 3.对各种was版本的兼容性较高 4.使用方便,官方自带 缺点也有很多: 1 ...

  3. SharePoint 2013 文档库“样式”变了

    有朋友反馈说文档库的样式变了. 经查证,原来有人修改了视图的"样式":库设置—视图—样式,改为默认即可. 另外,如果编辑页面,编辑web部件的属性,在"杂项"勾 ...

  4. oracle 11g 空表导出

    背景 oracle9用了一段时间,10用了一段时间,11现在算是主流了.11g也是坑人,空表竟然不导出,解决方法到时很多.这里只是记录下,知道有这个事情. 9的特点是还要用客户端管理工具链接服务器 1 ...

  5. 007_linux显示一个文件的某几行(中间几行)

    <1>从第3000行开始,显示1000行.即显示3000~3999行 cat -n filename | tail -n +3000 | head -n 1000 cat -n anaco ...

  6. 转载:Nginx是什么(1.1)《深入理解Nginx》(陶辉)

    原文:https://book.2cto.com/201304/19609.html 人们在了解新事物时,往往习惯通过类比来帮助自己理解事物的概貌.那么,我们在学习Nginx时也采用同样的方式,先来看 ...

  7. matplotlib画堆叠条形图

    import matplotlib.pyplot as plt%matplotlib inlineplt.style.use('ggplot') plt.style.use("ggplot& ...

  8. js对象深拷贝

    数组一维深拷贝:slice.concat.Array.from 对象一维深拷贝:Object.assign 一.利用扩展运算符...对数组中嵌套对象进行深拷贝 var arr=[{a:1,b:2},{ ...

  9. JQuery动画详解(四)

    一:基本动画show()显示隐藏的匹配元素.这个就是 'show( speed, [callback] )' 无动画的版本.如果选择的元素是可见的,这个方法将不会改变任何东西.无论这个元素是通过hid ...

  10. linux下安装openoffice

    一.环境 centos6.9 安装jdk1.6及以上 二.安装依赖 yum install libXext.x86_64 -y yum install freetype -y yum groupins ...