Description

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

Input

第一行为两个整数N,M。M是询问数,N是序列的长度(N<=100000,M<=200000)

第二行为N个整数,描述这个序列{ai},其中所有1<=ai<=N

再下面M行,每行两个整数x,y,

询问区间[l,r]由下列规则产生(OIER都知道是怎样的吧>_<):

l=min((x+lastans)mod n+1,(y+lastans)mod n+1);

r=max((x+lastans)mod n+1,(y+lastans)mod n+1);

Lastans表示上一个询问的答案,一开始lastans为0

Output

一共M行,每行给出每个询问的答案。

Sample Input

10 10
6 4 9 10 9 10 9 4 10 4
3 8
10 1
3 4
9 4
8 1
7 8
2 9
1 1
7 3
9 9

Sample Output

4
10
10
0
0
10
0
4
0
4

HINT

注意出题人为了方便,input的第二行最后多了个空格。

2015.6.24新加数据一组,2016.7.9放至40S,600M,但未重测

Source


  题目大意 询问区间内只出现1次数的最大值。强制在线。

  考虑每位置上的数会对哪些询问作出贡献。

  显然要让位置i对答案产生贡献询问的区间需要满足 pre[i] < l <= i 并且 i <= r < suf[i] 。

  那么就可以将每个询问(l, r)看成二维平面内的一个点。

  这样的话,每个位置对答案的贡献就可以通过2维线段树来进行维护。

  为了防止MLE,所以采取标记永久化的方案。但是注意修改操作是和线段树维护的信息取max。

Code

 /**
* bzoj
* Problem#3489
* Accepted
* Time: 21836ms
* Memory: 227780k
*/
#include <bits/stdc++.h>
using namespace std;
typedef bool boolean; typedef class SegTreeNode {
public:
SegTreeNode* val;
SegTreeNode *l, *r; SegTreeNode(int val = ):val((SegTreeNode*)val) { }
SegTreeNode(int val, SegTreeNode* org):val((SegTreeNode*)val) {
l = r = org;
} int intvalue() {
return (int)val;
} SegTreeNode*& pnodeval() {
return val;
} void set(int x) {
val = (SegTreeNode*)x;
}
}SegTreeNode; #define PLimit 10000000 //int alloced = 0;
SegTreeNode pool[PLimit];
SegTreeNode null = SegTreeNode(, &null);
SegTreeNode* top = pool; SegTreeNode* newnode() {
// alloced++;
if(top >= pool + PLimit)
return new SegTreeNode(, &null);
*top = SegTreeNode(, &null);
return top++;
} typedef class SegTree {
public:
int n;
SegTreeNode* rt; SegTree() { }
SegTree(int n):n(n), rt(newnode()) { } void update2(SegTreeNode*& node, int l, int r, int ql, int qr, int val) {
if(node == &null || node == NULL)
node = newnode();
if(l == ql && r == qr) {
if(val > node->intvalue())
node->set(val);
return;
}
int mid = (l + r) >> ;
if(ql <= mid)
update2(node->l, l, mid, ql, (qr > mid) ? (mid) : (qr), val);
if(qr > mid)
update2(node->r, mid + , r, (ql > mid) ? (ql) : (mid + ), qr, val);
} void update(SegTreeNode*& node, int l, int r, int ql, int qr, int qx, int qy, int val) {
if(node == &null)
node = newnode();
if(l == ql && r == qr) {
update2(node->pnodeval(), , n, qx, qy, val);
return;
}
int mid = (l + r) >> ;
if(ql <= mid)
update(node->l, l, mid, ql, (qr > mid) ? (mid) : (qr), qx, qy, val);
if(qr > mid)
update(node->r, mid + , r, (ql > mid) ? (ql) : (mid + ), qr, qx, qy, val);
} int query2(SegTreeNode*& node, int l, int r, int idx) {
if(node == NULL || node == &null)
return ;
if(l == idx && r == idx)
return node->intvalue();
int mid = (l + r) >> , rt = node->intvalue(), cmp = ;
if(idx <= mid)
cmp = query2(node->l, l, mid, idx);
else
cmp = query2(node->r, mid + , r, idx);
return (cmp > rt) ? (cmp) : (rt);
} int query(SegTreeNode*& node, int l, int r, int idx, int bidx) {
if(node == &null)
return ;
if(l == idx && r == idx)
return query2(node->pnodeval(), , n, bidx);
int mid = (l + r) >> , rt = query2(node->pnodeval(), , n, bidx), cmp;
if(idx <= mid)
cmp = query(node->l, l, mid, idx, bidx);
else
cmp = query(node->r, mid + , r, idx, bidx);
return (cmp > rt) ? (cmp) : (rt);
}
}SegTree; int n, m;
int *ar;
int *pre, *suf;
int *head;
SegTree st; inline void init() {
scanf("%d%d", &n, &m);
st = SegTree(n);
ar = new int[(n + )];
pre = new int[(n + )];
suf = new int[(n + )];
head = new int[(n + )];
for(int i = ; i <= n; i++)
scanf("%d", ar + i);
} inline void solve() {
memset(head, , sizeof(int) * (n + ));
for(int i = ; i <= n; i++)
pre[i] = head[ar[i]], head[ar[i]] = i;
fill(head, head + n + , n + );
for(int i = n; i; i--)
suf[i] = head[ar[i]], head[ar[i]] = i; for(int i = ; i <= n; i++)
st.update(st.rt, , n, pre[i] + , i, i, suf[i] - , ar[i]);//, fprintf(stderr, "%d: Memory Usage: %d (%d nodes)\n", i, alloced * sizeof(SegTreeNode), alloced);
int lastans = , x, y;
while(m--) {
scanf("%d%d", &x, &y);
x = (x + lastans) % n + ;
y = (y + lastans) % n + ;
if(x > y) swap(x, y);
lastans = st.query(st.rt, , n, x, y);
printf("%d\n", lastans);
}
} int main() {
init();
solve();
return ;
}

bzoj 3489 A simple rmq problem - 线段树的更多相关文章

  1. 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][ ...

  2. bzoj 3489 A simple rmq problem——主席树套线段树

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3489 题解:http://www.itdaan.com/blog/2017/11/24/9b ...

  3. BZOJ.3489.A simple rmq problem(主席树 Heap)

    题目链接 当时没用markdown写,可能看起来比较难受...可以复制到别的地方看比如DevC++. \(Description\) 给定一个长为n的序列,多次询问[l,r]中最大的只出现一次的数.强 ...

  4. bzoj 3489 A simple rmq problem —— 主席树套线段树

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3489 题解:http://www.itdaan.com/blog/2017/11/24/9b ...

  5. BZOJ 3489: A simple rmq problem

    3489: A simple rmq problem Time Limit: 40 Sec  Memory Limit: 600 MBSubmit: 1594  Solved: 520[Submit] ...

  6. BZOJ 3489 A simple rmq problem 可持久化KDtree/二维线段树

    题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=3489 题意概述: 给出一个序列,每次询问一个序列区间中仅出现了一次的数字最大是多少,如果 ...

  7. [BZOJ 3489] A simple rmq problem 【可持久化树套树】

    题目链接:BZOJ - 3489 题目分析 “因为是OJ上的题,就简单点好了.”——出题人 真的..好..简单... 首先,我们求出每个数的前一个与它相同的数的位置,即 prev[i] ,如果前面没有 ...

  8. BZOJ 3489 A simple rmq problem(可持久化线段树)

    题目链接:http://www.lydsy.com:808/JudgeOnline/problem.php?id=3489 题意:一个数列.每次询问一个区间内出现一次的最大的数字是多少. 思路:设la ...

  9. BZOJ 3489: A simple rmq problem (KD-tree做法)

    KD树水过这道可持久化树套树-其实就是个三维偏序 题解戳这里 CODE #include <bits/stdc++.h> using namespace std; #define ls ( ...

随机推荐

  1. dotnet 命令

    以下用实例串起dotnet常用命令,带你玩转dotnet命令. 1.创建(dotnet new) 首先我们创建一个项目,这里我们创建控制台程序,命令如下图所示. dotnet new dotnet n ...

  2. JavaScript原型继承的实例

    // 创建构造函数实例(获取DOM节点) <div id="app">测试字符</div>

  3. c# 在mongo中查询经纬度范围

    #region 索引 //IndexKeysDocument doc = new IndexKeysDocument();//新建索引 //2d 平面坐标索引,适用于基于平面的坐标计算.也支持球面距离 ...

  4. Attention Is All You Need 一些好的资料

    The encoders are all identical in structure (yet they do not share weights). Each one is broken down ...

  5. linux本地机上传文件到服务器

    最近工作全部切换到了linux环境下,就是吃喝拉撒全在linux下,微信,web端,qq,web端,-------,各种socket编程,网络通讯- 本地linux机从阿里云下载文件

  6. 【Linux学习五】文本处理

    环境 虚拟机:VMware 10 Linux版本:CentOS-6.5-x86_64 客户端:Xshell4 FTP:Xftp4 一.cut:显示切割的行数据f:选择显示的列s:不显示没有分隔符的行d ...

  7. mongodb查看操作记录方法以及用户添加删除权限修改密码

    前一阵跑程序时发现一个问题,同事导出了部分数据,但是在merge回原库时竟然和原库的数据对不上,后来找了半天发现是原库数据少了. 找了很多资料发现很多人认为的操作日志和我想的不太一样...找了半条才发 ...

  8. mongodb可视化工具 studio3t robo3T 下载安装使用介绍

    mongodb可视化工具 studio3t  robo3T 下载安装使用介绍 下载地址: https://studio3t.com/download robo3T

  9. golang学习笔记12 beego table name `xxx` repeat register, must be unique 错误问题

    golang学习笔记12 beego table name `xxx` repeat register, must be unique 错误问题 今天测试了重新建一个项目生成新的表,然后复制到旧的项目 ...

  10. 多进程wait、僵尸进程、孤儿进程、prctl

    1.概念 1.孤儿进程:一个父进程退出,而它的一个或多个子进程还在运行,那么那些子进程将成为孤儿进程.孤儿进程将被init进程(进程号为1)所收养,从而保证每个进程都会有一个父进程.而Init进程会自 ...