http://www.lydsy.com/JudgeOnline/problem.php?id=3489 (题目链接)

题意

  在线求区间不重复出现的最大的数。

Solution

  KDtree竟然能够处理这种问题,好神啊。

  以下转自:http://trinklee.blog.163.com/blog/static/2381580602015422933539/

  记录每个位置的数前一次出现的位置pre[i]和后一次出现的位置nxt[i],然后我们询问的就是

  1. l<=i<=r

  2. pre[i]<l

  3. nxt[i]>r

  满足三个条件下的max(a[i])

  将每个点的信息看作三维空间上带权值的点(i,pre[i],nxt[i]),然后建立kdtree。

  询问的话,等价于第一维在[l,r]范围内,第二维在[0,l-1]范围内,第三维在[r+1,+oo]范围内的一个三维空间内,查询在里面的点权最大值。于是这样就能转换成kdtree啦~

  关于kdtree:

  1. 建树跟二维的一样建法,xyz三个坐标轮流换,并且维护当前域内的点权max

  2. 查询的时候,如果当前域内max<=ans,直接不做(剪枝1),如果当前点在查询域内则更新答案,如果子空间与查询域不交则不查(剪枝2)

细节

  竟然1A了w(゚Д゚)w

代码

// bzoj3489
#include<algorithm>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<ctime>
#define LL long long
#define inf 1<<30
#define Pi acos(-1.0)
#define free(a) freopen(a".in","r",stdin),freopen(a".out","w",stdout);
using namespace std; const int maxn=100010,maxm=200010;
int a[maxn],head[maxn],pre[maxn],nxt[maxn],ax[3],ay[3];
int D,n,m,rt; struct KDtree {
int l,r,val,Max,v[3],mn[3],mx[3];
friend bool operator < (const KDtree a,const KDtree b) {
return a.v[D]<b.v[D];
}
}tr[maxn]; void update(int k) {
for (int i=0;i<=2;i++) {
if (tr[k].l) {
tr[k].mx[i]=max(tr[k].mx[i],tr[tr[k].l].mx[i]);
tr[k].mn[i]=min(tr[k].mn[i],tr[tr[k].l].mn[i]);
}
if (tr[k].r) {
tr[k].mx[i]=max(tr[k].mx[i],tr[tr[k].r].mx[i]);
tr[k].mn[i]=min(tr[k].mn[i],tr[tr[k].r].mn[i]);
}
}
if (tr[k].l) tr[k].Max=max(tr[k].Max,tr[tr[k].l].Max);
if (tr[k].r) tr[k].Max=max(tr[k].Max,tr[tr[k].r].Max);
}
int build(int l,int r,int p) {
D=p;
int mid=(l+r)>>1;
nth_element(tr+l,tr+mid,tr+r+1);
if (l<mid) tr[mid].l=build(l,mid-1,(p+1)%3);
if (r>mid) tr[mid].r=build(mid+1,r,(p+1)%3);
update(mid);
return mid;
}
bool in(int x,int y,int X,int Y) {
return x>=X && y<=Y;
}
bool out(int x,int y,int X,int Y) {
return y<X || x>Y;
}
int query(int k) {
if (!k) return 0;
int flag=1,res=0;
for (int i=0;i<=2;i++) {
if (out(tr[k].mn[i],tr[k].mx[i],ax[i],ay[i])) return 0;
flag&=in(tr[k].mn[i],tr[k].mx[i],ax[i],ay[i]);
}
if (flag) return tr[k].Max; flag=1;for (int i=0;i<=2;i++) flag&=in(tr[k].v[i],tr[k].v[i],ax[i],ay[i]);
if (flag) res=tr[k].val; if (tr[tr[k].l].Max>tr[tr[k].r].Max) {
if (res<tr[tr[k].l].Max) res=max(res,query(tr[k].l));
if (res<tr[tr[k].r].Max) res=max(res,query(tr[k].r));
}
else {
if (res<tr[tr[k].r].Max) res=max(res,query(tr[k].r));
if (res<tr[tr[k].l].Max) res=max(res,query(tr[k].l));
}
return res;
} int main() {
scanf("%d%d",&n,&m);
for (int i=1;i<=n;i++) {
scanf("%d",&a[i]);
int x=head[a[i]];head[a[i]]=i;
pre[i]=x;nxt[x]=i;
}
for (int i=1;i<=n;i++) {
tr[i].v[0]=tr[i].mx[0]=tr[i].mn[0]=i;
tr[i].v[1]=tr[i].mx[1]=tr[i].mn[1]=pre[i];
tr[i].v[2]=tr[i].mx[2]=tr[i].mn[2]=nxt[i] ? nxt[i] : n+1;
tr[i].val=tr[i].Max=a[i];
}
rt=build(1,n,0);
int ans=0;
for (int l,r,x,y,i=1;i<=m;i++) {
scanf("%d%d",&x,&y);
l=min((x+ans)%n+1,(y+ans)%n+1);
r=max((x+ans)%n+1,(y+ans)%n+1);
ax[0]=l,ay[0]=r;
ax[1]=0,ay[1]=l-1;
ax[2]=r+1,ay[2]=n+1;
ans=query(rt);
printf("%d\n",ans);
}
return 0;
}

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

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

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

  2. 【BZOJ3489】A simple rmq problem

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

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

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

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

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

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

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

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

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

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

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

  8. 【bzoj 3489】A simple rmq problem

    题目 \(kdt\)就是数点神器 我们先扫两遍处理出每个数上一次出现的位置\(pre_i,nxt_i\),之后变成\((i,pre_i,nxt_i)\)这样一个三维空间上的点 就变成了求一个立方体的最 ...

  9. BZOJ3489:A simple rmq problem

    浅谈\(K-D\) \(Tree\):https://www.cnblogs.com/AKMer/p/10387266.html 题目传送门:https://lydsy.com/JudgeOnline ...

随机推荐

  1. Eclipse安装Git插件(在线和离线)

    在线安装: help-->install new software-->add location就是安装的地址:http://download.eclipse.org/egit/updat ...

  2. 大数据入门第十二天——flume入门

    一.概述 1.什么是flume 官网的介绍:http://flume.apache.org/ Flume is a distributed, reliable, and available servi ...

  3. 2017-2018-1 20155331 嵌入式C语言

    2017-2018-1 20155331 嵌入式C语言 作业要求: 在作业本上完成附图作业,要认真看题目要求. 提交作业截图 作弊本学期成绩清零(有雷同的,不管是给别人传答案,还是找别人要答案都清零) ...

  4. 变量内存空间的释放---c语言

    堆栈内存释放: 栈的内存是由编译器自动分配.释放,出了作用域就释放. 堆的内存由程序员分配.释放,他的作用域是整个程序,如果程序没有释放,程序结束时会自动释放.

  5. python 回溯法 子集树模板 系列 —— 13、最佳作业调度问题

    问题 给定 n 个作业,每一个作业都有两项子任务需要分别在两台机器上完成.每一个作业必须先由机器1 处理,然后由机器2处理. 试设计一个算法找出完成这n个任务的最佳调度,使其机器2完成各作业时间之和达 ...

  6. Eclipse中Hadoop插件配置

    Eclipse中Hadoop插件DFS配置 http://www.cnblogs.com/xia520pi/archive/2012/05/20/2510723.html

  7. JavaScript快速入门-简介

    一.JavaScript历史(摘自w3school) JavaScript 是因特网上最流行的脚本语言,它存在于全世界所有 Web 浏览器中,能够增强用户与 Web 站点和 Web 应用程序之间的交互 ...

  8. nodejs安装及npm模块插件安装路径配置

    在学习完js后,我们就要进入nodejs的学习,因此就必须配置nodejs和npm的属性了. 我相信,个别人在安装时会遇到这样那样的问题,看着同学都已装好,难免会焦虑起来.于是就开始上网查找解决方案, ...

  9. CodeMirror 小册子

    User manual and reference guide      version 5.41.1 用户手册和参考指南 CodeMirror is a code-editor component ...

  10. 12.18daily_scrum

    本软件中的最后一个界面——“关于”界面的设计已经开始进行,数据传输的内容也差不多进行过半,最主要的任务依旧在测试过程中,我们组接下来还是要大力加强测试的强度和数量,注意边际数据和错误数据的测试处理: ...