题目

\(kdt\)就是数点神器

我们先扫两遍处理出每个数上一次出现的位置\(pre_i,nxt_i\),之后变成\((i,pre_i,nxt_i)\)这样一个三维空间上的点

就变成了求一个立方体的最大值了

随便剪剪枝就过了

代码

#include<bits/stdc++.h>
#define re register
#define max(a,b) ((a)>(b)?(a):(b))
inline int read() {
char c=getchar();int x=0;while(c<'0'||c>'9') c=getchar();
while(c>='0'&&c<='9') x=(x<<3)+(x<<1)+c-48,c=getchar();return x;
}
const int maxn=100005;
struct Point{int v,x[3];}p[maxn];
int a[maxn],pre[maxn],nxt[maxn],st[maxn];
int n,m,op,L[3],R[3],cnt,lst;
int mn[maxn][3],mx[maxn][3],pos[maxn],l[maxn],r[maxn],d[maxn];
inline int cmp(const Point &A,const Point &B) {return A.x[op]<B.x[op];}
inline void pushup(int k) {
mx[k][0]=mn[k][0]=p[pos[k]].x[0];
mx[k][1]=mn[k][1]=p[pos[k]].x[1];
mx[k][2]=mn[k][2]=p[pos[k]].x[2];
d[k]=max(d[l[k]],d[r[k]]);d[k]=max(d[k],p[pos[k]].v);
for(re int i=0;i<3;i++) {
if(mx[l[k]][i]>mx[k][i]) mx[k][i]=mx[l[k]][i];
if(mx[r[k]][i]>mx[k][i]) mx[k][i]=mx[r[k]][i];
if(l[k]&&mn[l[k]][i]<mn[k][i]) mn[k][i]=mn[l[k]][i];
if(r[k]&&mn[r[k]][i]<mn[k][i]) mn[k][i]=mn[r[k]][i];
}
}
int build(int x,int y,int o) {
if(x>y) return 0;
int mid=x+y>>1,k=++cnt;
op=o%3;std::nth_element(p+x,p+mid,p+y+1,cmp);
l[k]=build(x,mid-1,o+1),r[k]=build(mid+1,y,o+1);
pos[k]=mid;pushup(k);return k;
}
inline int chk(int k) {
return p[k].x[0]>=L[0]&&p[k].x[0]<=R[0]&&p[k].x[1]>=L[1]&&p[k].x[1]<=R[1]&&p[k].x[2]>=L[2]&&p[k].x[2]<=R[2];
}
inline int out(int k) {
return R[0]<mn[k][0]||L[0]>mx[k][0]||R[1]<mn[k][1]||L[1]>mx[k][1]||R[2]<mn[k][2]||L[2]>mx[k][2];
}
inline int in(int k) {
return L[0]<=mn[k][0]&&R[0]>=mx[k][0]&&L[1]<=mn[k][1]&&R[1]>=mx[k][1]&&L[2]<=mn[k][2]&&R[2]>=mx[k][2];
}
void query(int k) {
if(in(k)) {lst=max(lst,d[k]);return;}
if(chk(pos[k])) lst=max(lst,p[pos[k]].v);
if(l[k]&&!out(l[k])&&lst<d[l[k]]) query(l[k]);
if(r[k]&&!out(r[k])&&lst<d[r[k]]) query(r[k]);
}
int main() {
n=read(),m=read();
for(re int i=1;i<=n;i++) a[i]=read();
for(re int i=1;i<=n;i++)
pre[i]=st[a[i]],st[a[i]]=i;
for(re int i=1;i<=n;i++) st[i]=n+1;
for(re int i=n;i;--i)
nxt[i]=st[a[i]],st[a[i]]=i;
for(re int i=1;i<=n;i++)
p[i].x[0]=i,p[i].x[1]=pre[i],p[i].x[2]=nxt[i],p[i].v=a[i];
build(1,n,0);int x,y;
L[1]=0,R[2]=n+1;
while(m--) {
x=read()+lst,y=read()+lst;
x%=n,y%=n,++x,++y;
if(x>y) std::swap(x,y);
L[0]=x,R[0]=y;R[1]=x-1,L[2]=y+1;
lst=0;query(1);printf("%d\n",lst);
}
return 0;
}

【bzoj 3489】A simple rmq problem的更多相关文章

  1. 【BZOJ】【3489】A simple rmq problem

    KD-Tree(乱搞) Orz zyf教给蒟蒻做法 蒟蒻并不会这题正解……(可持久化树套树?...Orz 对于每个点,我们可以求出pre[i],nex[i],那么询问的答案就是:求max (a[i]) ...

  2. 【BZOJ3489】A simple rmq problem(KD-Tree)

    [BZOJ3489]A simple rmq problem(KD-Tree) 题面 BZOJ 题解 直接做肯定不好做,首先我们知道我们是一个二维平面数点,但是限制区间只能出现一次很不好办,那么我们给 ...

  3. 【BZOJ3489】A simple rmq problem

    [BZOJ3489]A simple rmq problem 题面 bzoj 题解 这个题不强制在线的话随便做啊... 考虑强制在线时怎么搞 预处理出一个位置上一个出现的相同数的位置\(pre\)与下 ...

  4. 【BZOJ3489】A simple rmq problem kd-tree

    [BZOJ3489]A simple rmq problem Description 因为是OJ上的题,就简单点好了.给出一个长度为n的序列,给出M个询问:在[l,r]之间找到一个在这个区间里只出现过 ...

  5. 【bzoj3489】 A simple rmq problem

    http://www.lydsy.com/JudgeOnline/problem.php?id=3489 (题目链接) 题意 在线求区间不重复出现的最大的数. Solution KDtree竟然能够处 ...

  6. 【bzoj3489】 A simple rmq problem k-d树

    由于某些原因,我先打了一个错误的树套树,后来打起了$k-d$.接着因不明原因在思路上被卡了很久,在今天中午蹲坑时恍然大悟...... 对于一个数字$a_i$,我们可以用一组三维坐标$(i,pre,nx ...

  7. 【BZOJ3489】A simple rmq problem【kd树】

    题意 给出一个长度为n的序列,给出M个询问:在[l,r]之间找到一个在这个区间里只出现过一次的数,并且要求找的这个数尽可能大.如果找不到这样的数,则直接输出0.我会采取一些措施强制在线. 分析 预处理 ...

  8. 【bzoj3489】A simple rmq problem 三维KD-tree

    题目描述 因为是OJ上的题,就简单点好了.给出一个长度为n的序列,给出M个询问:在[l,r]之间找到一个在这个区间里只出现过一次的数,并且要求找的这个数尽可能大.如果找不到这样的数,则直接输出0.我会 ...

  9. 【BZOJ 3218】 3218: a + b Problem(最小割+可持久化线段树)

    3218: a + b Problem Time Limit: 20 Sec  Memory Limit: 40 MBSubmit: 1440  Solved: 545 Description Inp ...

随机推荐

  1. 【JZOJ6285】飘雪圣域

    description analysis 从求联通块出发根本没做法,于是考虑连通块里面的边 对于一个询问\([l,r]\),一条边的左端点\(≥l\)且右端点\(≤r\)才在这个区间的点之间 于是对于 ...

  2. JUC 一 CyclicBarrier 与 Semaphore

    java.util.concurrent CyclicBarrier简介 CyclicBarrier:可重用屏障/栅栏 类似于 CountDownLatch(倒计数闭锁),它能阻塞一组线程直到某个事件 ...

  3. PHP headers_sent() 函数

    定义和用法 headers_sent() 函数检查 HTTP 报头是否发送/已发送到何处. 如果报头已发送,该函数返回 TRUE,否则返回 FALSE. 语法 headers_sent(file,li ...

  4. 38 ubuntu/windows双系统安装

    0 引言 (1)针对bios 和 uefi引导,安装方式略有不同. (2)针对nvidia显卡,在安装时需要特殊设置. 1 EasyBCD安装方式介绍-适用于bios引导方式 参考百度经验贴安装即可, ...

  5. DELPHI 数据库操作

    DELPHI 把数据库中的数据转换成XML格式 function ReplaceString(AString: string): string; begin Result := StringRepla ...

  6. NX二次开发-NXOPEN_DimensionCollection遍历图纸上的所有标注尺寸

    NX11+VS2013 #include <NXOpen/Drawings_DrawingSheet.hxx> #include <NXOpen/Drawings_DrawingSh ...

  7. NX二次开发-创建圆弧(圆心-半径)UF_CURVE_create_arc_center_radius

    NX9+VS2012 #include <uf.h> #include <uf_curve.h> #include <uf_modl.h> UF_initializ ...

  8. NX二次开发-相对路径环境变量和绝对路径环境变量

    相对路径环境变量:${UGII_BASE_DIR}\CaesarToolkits 绝对路径环境变量:D:\Program Files\Siemens\NX 9.0\CaesarToolkits

  9. NX二次开发-关闭信息窗口UF_UI_close_listing_window

    #include <uf.h> #include <uf_ui.h> UF_initialize(); //打开信息窗口 UF_UI_open_listing_window() ...

  10. 使用Socket.IO做单页SPA应用更新

    单页应用的挑战之一是确保客户端软件和服务器应用相匹配. 举例:如果一个用Bobbie在他的浏览器中加载我们的单页应用,五分钟之后我们更新了服务器应用.现在Bobbiede遇到了问题,因为我们对服务器做 ...