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. 46.HTML---18个学习 flexbox 的优质资源

    大家都在谈论的“flexbox”是什么呢?它能帮我们解决什么问题呢?现在你可以在你的项目中使用“flexbox”吗?这些指南.教程.网站和工具会告诉你你需要知道的关于“flexbox”的知识. Lea ...

  2. ecshop 订单状态

    ecshop的订单状态都是在ecs_order_info表中的字段里. 订单状态 未确认 取消 确认 已付款 配货中 已发货 已收货 退货 order_status 0 2 1 1 1 5 5 4 s ...

  3. 用C#创建一个窗体,在构造函数里面写代码和在from_load事件里面写代码有什么不同?

    没太大区别.一区别就是代码加载时间先后的问题.构造函数先加载,load事件中后加载.

  4. python 正则基本方法

    2017-04-11 学习python,免不了应对爬虫,初学爬虫最难理解的就是正则表达式. 比如我们要爬去网页上的某些内容,就像下面这种形式: <p>safdsf</p>< ...

  5. 4.构造Thread对象你也许不知道的几件事

    1.Thread类对象只有在调用了start()方法之后,JVM虚拟机才会给我们创建一个真正的线程!否则就不能说是创建了线程!也就是说new Thread()之后,此时实际上在计算机底层,操作系统实际 ...

  6. STL算法中函数对象和谓词

    函数对象和谓词定义 函数对象: 重载函数调用操作符的类,其对象常称为函数对象(function object),即它们是行为类似函数的对象.一个类对象,表现出一个函数的特征,就是通过“对象名+(参数列 ...

  7. Yii Restful api自定义字段

  8. sitecore系统教程之内容编辑器中创建项目

    在内容编辑器中创建新项目时,必须先在内容树中选择一个项目,以指示新项目的位置.您可以创建一个新项目作为您选择的项目的兄弟或子项目: 兄弟是您在与所选项目相同的级别创建的项目. 子项是您在所选项下创建的 ...

  9. codeforces 979A Pizza, Pizza, Pizza!!!

    题意: 对一个圆形的pizza,只能用直线来切它,求把它切为n+1份的形状和size都相同的最下次数. 思路: 形状和size都相同,那么只能是扇形,分奇偶讨论. n为0还得特判,切0刀,因为这个还被 ...

  10. MVC请求管道

    下面是请求管道中的19个事件. (1)BeginRequest: 开始处理请求 (2)AuthenticateRequest授权验证请求,获取用户授权信息 (3):PostAuthenticateRe ...