考前写写板子。

用$(i,pre[i],nxt[i])$来描述一个点,然后就变成了区间求最值的问题。

KD-Tree 由低维转向高维的方法,可以用来敲暴力。

剩下就是KD-Tree的基本操作了。

#include <map>
#include <cmath>
#include <queue>
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
#define F(i,j,k) for (int i=j;i<=k;++i)
#define D(i,j,k) for (int i=j;i>=k;--i)
#define inf 0x3f3f3f3f
#define L t[o].ch[0]
#define R t[o].ch[1]
#define ll long long
#define mp make_pair
#define maxn 200005 int D,a[maxn],n,m,ans,pre[maxn],nxt[maxn],lst[maxn],rt;
int ql,qr;
struct Point{
int d[3],mn[3],mx[3],ch[2],v,vmax;
}t[maxn];//i pre nxt bool operator < (Point a,Point b){return a.d[D]<b.d[D];} void pushup(int o)
{
F(i,0,2) t[o].mx[i]=t[o].mn[i]=t[o].d[i];
F(i,0,2)
{
t[o].mx[i]=max(max(t[L].mx[i],t[R].mx[i]),t[o].mx[i]);
t[o].mn[i]=min(min(t[L].mn[i],t[R].mn[i]),t[o].mn[i]);
}
t[o].vmax=max(t[o].v,max(t[L].vmax,t[R].vmax));
} void init(){F(i,0,2) t[0].mn[i]=inf,t[0].mx[i]=-inf;} int build(int l,int r,int dir)
{
D=dir;int mid=l+r>>1;int o=mid;
nth_element(t+l,t+mid,t+r+1);
L=l<mid?build(l,mid-1,(dir+1)%3):0;
R=r>mid?build(mid+1,r,(dir+1)%3):0;
pushup(o); return o;
} bool in(int o)
{
if (t[o].mx[1]>=ql||t[o].mn[2]<=qr) return 0;
if (t[o].mx[0]>qr||t[o].mx[1]<ql) return 0;
return 1;
} bool pin(int o)
{
if (t[o].d[0]>=ql&&t[o].d[0]<=qr&&t[o].d[1]<ql&&t[o].d[2]>qr) return 1;
return 0;
} bool check(int o)
{
if (t[o].mx[0]<ql||t[o].mn[0]>qr) return 0;
if (t[o].mn[1]>=ql||t[o].mx[2]<=qr) return 0;
return 1;
} void query(int o)
{
if (!o) return;
if (in(o)) {ans=max(ans,t[o].vmax);return;}
if (pin(o)) ans=max(ans,t[o].v);
if (t[L].vmax>t[R].vmax)
{
if (t[L].vmax>ans&&check(L)) query(L);
if (t[R].vmax>ans&&check(R)) query(R);
}
else
{
if (t[R].vmax>ans&&check(R)) query(R);
if (t[L].vmax>ans&&check(L)) query(L);
}
} int main()
{
scanf("%d%d",&n,&m);init();
F(i,1,n) scanf("%d",&a[i]);
F(i,1,n) lst[i]=0;
F(i,1,n) pre[i]=lst[a[i]],lst[a[i]]=i;
F(i,1,n) lst[i]=n+1;
D(i,n,1) nxt[i]=lst[a[i]],lst[a[i]]=i;
F(i,1,n) t[i].d[0]=i,t[i].d[1]=pre[i],t[i].d[2]=nxt[i],t[i].v=a[i];
rt=build(1,n,0);
F(i,1,m)
{
int x,y;
scanf("%d%d",&x,&y);
ql=min((x+ans)%n+1,(y+ans)%n+1);
qr=max((x+ans)%n+1,(y+ans)%n+1);
ans=0;query(rt);printf("%d\n",ans);
}
return 0;
}

  

BZOJ 3489 A simple rmq problem ——KD-Tree的更多相关文章

  1. BZOJ 3489: A simple rmq problem(K-D Tree)

    Time Limit: 40 Sec  Memory Limit: 512 MBSubmit: 2579  Solved: 888[Submit][Status][Discuss] Descripti ...

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

  3. BZOJ 3489: A simple rmq problem

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

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

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

  5. BZOJ3489 A simple rmq problem K-D Tree

    传送门 什么可持久化树套树才不会写呢,K-D Tree大法吼啊 对于第\(i\)个数,设其前面最后的与它值相同的位置为\(pre_i\),其后面最前的与它值相同的位置为\(aft_i\),那么对于一个 ...

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

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

  7. bzoj 3489 A simple rmq problem - 线段树

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

  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 可持久化KDtree/二维线段树

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

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

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

随机推荐

  1. UVA 11997 K Smallest Sums (多路归并)

    从包含k个整数的k个数组中各选一个求和,在所有的和中选最小的k个值. 思路是多路归并,对于两个长度为k的有序表按一定顺序选两个数字组成和,(B表已经有序)会形成n个有序表 A1+B1<=A1+B ...

  2. 2018.4.1 Ubuntu16.04 下配置Tomcat服务器以及设置dingshi启动

    Tomcat自启动的设置技巧 以root用户登录系统: cd /etc/rc.d/init.d/ vi tomcat #!/bin/sh # # tomcat: Start/Stop/Restart ...

  3. python_87_shelve模块

    'shelve模块是一个简单的key,value将内存数据通过文件持久化的模块,可以持久化任何pickle可支持的python数据格式(只支持pickle)' #序列化,将数据写入文件 import ...

  4. Mysql中反引号和单引号的区别

    反引号,一般在ESC键的下方. 它是为了区分MYSQL的保留字与普通字符而引入的符号.举个例子:SELECT `select` FROM `test` WHERE select='字段值'在test表 ...

  5. jenkins 全局工具配置

  6. python入门:输出1-10的所有数(自写)

    #!/usr/bin/env python # -*- coding:utf-8 -*- #输出1-10的所有数(自写) """ 导入time库,给kaishi赋值为数字 ...

  7. 基于Centos7.2使用Cobbler工具定制化批量安装Centos7.2系统

    1.1    定制Centos_7_x86_64.ks文件内容 # Cobbler for Kickstart Configurator for CentOS 7.2.1511 by Wolf_Dre ...

  8. zookeeper伪集群(一)

    Zookeeper的安装和配置十分简单, 既可以配置成单机模式, 也可以配置成伪集群模式.集群模式. 本人将对伪集群.集群进行重点介绍: 铺垫: 1.集群必须是奇数(2N+1),伪集群和集群一致. 2 ...

  9. leetcode-11-dfs

    DFS算法: explore(G, v) visited(v) = trueprevisit(v) for each edge(v, u) in E: if not visited(u): explo ...

  10. HDU - 4763 Theme Section (KMP的next数组的应用)

    给定一个字符串,求出一个前缀A,使得字符串的构成可以表示成ABABA的形式(B可以为空串). 输出这个前缀的最大长度. KMP算法Next数组的使用. 枚举中间的每个位置,可以根据Next数组求出这个 ...