思路:

1.找次大值 我们不妨设当前点是次大的 那这段区间为 左边第二个比它大的点的坐标+1 和右边第二个比它大的点的坐标-1

2.用可持久化trie树找异或最大值

也可以用莫队

//By SiriusRen
#include <set>
#include <cstdio>
#include <algorithm>
using namespace std;
#define N 50050
int n,cnt,f,ch[N*32][2],wei[N*32],root[N];
int begin,end,ans;
void insert(int last,int num){
int now=cnt;
for(int i=30;i>=0;i--){
f=num&(1<<i)?1:0;
ch[now][f]=++cnt,ch[now][!f]=ch[last][!f];
now=ch[now][f],last=ch[last][f];
wei[now]=wei[last]+1;
}
}
int query(int x,int y,int num){
int temp=0;
for(int i=30;i>=0;i--){
f=num&(1<<i)?0:1;
if(wei[ch[y][f]]-wei[ch[x][f]]>0)
x=ch[x][f],y=ch[y][f],temp+=(1<<i);
else x=ch[x][!f],y=ch[y][!f];
}return temp;
}
struct Node{int id,w;}a[N];
bool operator<(Node a,Node b){return a.w>b.w;}
set<int>st;
set<int>::iterator it1,it2;
int main(){
scanf("%d",&n);
for(int i=1;i<=n;i++){
scanf("%d",&a[i].w),a[i].id=i;
root[i]=++cnt;insert(root[i-1],a[i].w);
}
st.insert(-1),st.insert(-2),st.insert(N),st.insert(n+1);
sort(a+1,a+1+n),st.insert(a[1].id);
for(int i=2;i<=n;i++){
st.insert(a[i].id);
it2=st.lower_bound(a[i].id);
it1=it2,it2++,it2++,it1--,it1--;
begin=max(1,*it1+1),end=min(n,*it2-1);
ans=max(ans,query(root[begin-1],root[end],a[i].w));
}
printf("%d\n",ans);
}

莫队(是稍微慢一些的……):

//By SiriusRen
#include <set>
#include <cstdio>
#include <algorithm>
using namespace std;
#define N 50050
int n,cnt=1,b[N],f,ch[N*32][2],wei[N*32],root[N];
int begin,end,ans,tot;
struct Node{int id,w;}a[N];
bool operator<(Node a,Node b){return a.w>b.w;}
set<int>st;
set<int>::iterator it1,it2;
struct Ask{int l,r,w;}ask[N];
bool operator<(Ask a,Ask b){return a.l<b.l;}
void insert(int x,int w){
int now=1;
for(int i=30;i>=0;i--){
f=x&(1<<i)?1:0;
if(!ch[now][f])ch[now][f]=++cnt;
now=ch[now][f],wei[now]+=w;
}
}
int query(int num){
int now=1,temp=0;
for(int i=30;i>=0;i--){
f=num&(1<<i)?0:1;
if(ch[now][f]&&wei[ch[now][f]])temp+=(1<<i),now=ch[now][f];
else now=ch[now][!f];
}return temp;
}
int main(){
scanf("%d",&n);
for(int i=1;i<=n;i++)scanf("%d",&b[i]),a[i].w=b[i],a[i].id=i;
st.insert(-1),st.insert(-2),st.insert(N),st.insert(n+1);
sort(a+1,a+1+n),st.insert(a[1].id);
for(int i=2;i<=n;i++){
st.insert(a[i].id);
it2=st.lower_bound(a[i].id);
it1=it2,it2++,it2++,it1--,it1--;
begin=max(1,*it1+1),end=min(n,*it2-1);
ask[++tot].l=begin-1,ask[tot].r=end,ask[tot].w=a[i].w;
}
sort(ask+1,ask+1+tot);
for(int i=1,l=1,r=0;i<=tot;i++){
while(r<ask[i].r)insert(b[r+1],1),r++;
while(r>ask[i].r)insert(b[r],-1),r--;
while(l<ask[i].l)insert(b[l],-1),l++;
while(l>ask[i].l)insert(b[l-1],1),l--;
ans=max(ans,query(ask[i].w));
}
printf("%d\n",ans);
}

BZOJ 3166 set+可持久化trie树(OR 莫队)的更多相关文章

  1. 主席树||可持久化线段树+离散化 || 莫队+分块 ||BZOJ 3585: mex || Luogu P4137 Rmq Problem / mex

    题面:Rmq Problem / mex 题解: 先离散化,然后插一堆空白,大体就是如果(对于以a.data<b.data排序后的A)A[i-1].data+1!=A[i].data,则插一个空 ...

  2. bzoj 3261: 最大异或和 (可持久化trie树)

    3261: 最大异或和 Time Limit: 10 Sec  Memory Limit: 512 MB Description       给定一个非负整数序列 {a},初始长度为 N.       ...

  3. BZOJ 3261 最大异或和 可持久化Trie树

    题目大意:给定一个序列,提供下列操作: 1.在数组结尾插入一个数 2.给定l,r,x,求一个l<=p<=r,使x^a[p]^a[p+1]^...^a[n]最大 首先我们能够维护前缀和 然后 ...

  4. [BZOJ3261&BZOJ3166]可持久化trie树及其应用

    可持久化trie树 可持久化trie树现在想来是比较好理解的了,但却看了一个下午... 相当于对于每个状态建立一条链(或者说一棵trie),求解的时候只要让两个点按照相同的步子走然后看sum的大小关系 ...

  5. 可持久化Trie树初步

    可持久化Trie树和可持久化线段树很像,依次插入信息,通过减法来进行历史版本查询. 2015年11月27日 bzoj3261 最大异或和 我们需要计算 a[p] xor a[p+1] xor ... ...

  6. [十二省联考2019]异或粽子——可持久化trie树+堆

    题目链接: [十二省联考2019]异或粽子 求前$k$大异或区间,可以发现$k$比较小,我们考虑找出每个区间. 为了快速得到一个区间的异或和,将原序列做前缀异或和. 对于每个点作为右端点时,我们维护出 ...

  7. BZOJ4477[Jsoi2015]字符串树——可持久化trie树

    题目描述 萌萌买了一颗字符串树的种子,春天种下去以后夏天就能长出一棵很大的字符串树.字符串树很奇特,树枝上都密密麻麻写满了字符串,看上去很复杂的样子.[问题描述]字符串树本质上还是一棵树,即N个节点N ...

  8. BZOJ5338 [TJOI2018] Xor 【可持久化Trie树】【dfs序】

    题目分析: 很无聊的一道题目.首先区间内单点对应异或值的询问容易想到trie树.由于题目在树上进行,case1将路径分成两段,然后dfs的时候顺便可持久化trie树做询问.case2维护dfs序,对d ...

  9. 51nod 1295 XOR key (可持久化Trie树)

    1295 XOR key  题目来源: HackerRank 基准时间限制:1.5 秒 空间限制:262144 KB 分值: 160 难度:6级算法题   给出一个长度为N的正整数数组A,再给出Q个查 ...

随机推荐

  1. hdoj--1028--Ignatius and the Princess III(母函数)

    Ignatius and the Princess III Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K ...

  2. Linux操作系统下Oracle主要监控工具介绍

    Oracle监控包括有效且完全地监控Oracle数据库的性能.可用性和使用率等统计量,还包括即时的错误通知和纠正措施,并提供全面的报表和图表.本文中主要介绍几种Linux操作系统下Oracle主要监控 ...

  3. C#读出文本文件内容,遍历数组筛选出 含有汉字对应的拼音字符

    情景描述:由于任务需要,现有一用户表数据,用户名 字段 在新增用户时,输入中文和拼音两种,先要区分同时含有中文和拼音字母的用户名.由于数据很多,可以通过一段代码完成查询: 前提:在阅读本文之前可以先了 ...

  4. No changes detected or App 'blog' could not be found. Is it in INSTALLED_APPS?

    出现该问题的原因: django没有在setting.py的配置文件中找到app内容,需要增加app的名称 E:\PycharmProjects\Mysite>python manage.py ...

  5. kubernetes系列:(一)、kubeadm搭建kubernetes(v1.13.1)单节点集群

    kubeadm是Kubernetes官方提供的用于快速部署Kubernetes集群的工具,本篇文章使用kubeadm搭建一个单master节点的k8s集群. 节点部署信息 节点主机名 节点IP 节点角 ...

  6. Unity 烘焙的2种方式

    游戏场景通常有许多光源,使用实时渲染会非常消耗性能,解决办法是烘焙,烘焙有2种方式. 1, 在3dmax等模型制作软件中对场景进行烘焙.将烘焙好的模型以及贴图导入到unity3d. 相对复杂但效果好 ...

  7. JavaScript中的线程与进程

    定义: 线程分为:单线程和多线程 单线程:一个正在运行的程序(即进行)至少有一个线程,这个线程叫做主线程,只有一个主线程的程序叫做单线程程序,主线程负责执行所有代码的执行(UI展现及刷新.网络请求.本 ...

  8. activity工作流学习地址

    https://wenku.baidu.com/view/8572153150e2524de4187e5d.html

  9. [读书笔记] Python数据分析 (一) 准备工作

    1. python中数据结构:矩阵,数组,数据框,通过关键列相互联系的多个表(SQL主键,外键),时间序列 2. python 解释型语言,程序员时间和CPU时间衡量,高频交易系统 3. 全局解释器锁 ...

  10. Python socket通信之FTP

    Python中利用socket进行server端和client端通信是网络编程的基础,是最简单的传输范例. (懂网络的请自动跳过这一部分) 首先,要想通信,必须建立连接,建立连接的过程,需要clien ...