BZOJ 3166 set+可持久化trie树(OR 莫队)
思路:
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 莫队)的更多相关文章
- 主席树||可持久化线段树+离散化 || 莫队+分块 ||BZOJ 3585: mex || Luogu P4137 Rmq Problem / mex
题面:Rmq Problem / mex 题解: 先离散化,然后插一堆空白,大体就是如果(对于以a.data<b.data排序后的A)A[i-1].data+1!=A[i].data,则插一个空 ...
- bzoj 3261: 最大异或和 (可持久化trie树)
3261: 最大异或和 Time Limit: 10 Sec Memory Limit: 512 MB Description 给定一个非负整数序列 {a},初始长度为 N. ...
- BZOJ 3261 最大异或和 可持久化Trie树
题目大意:给定一个序列,提供下列操作: 1.在数组结尾插入一个数 2.给定l,r,x,求一个l<=p<=r,使x^a[p]^a[p+1]^...^a[n]最大 首先我们能够维护前缀和 然后 ...
- [BZOJ3261&BZOJ3166]可持久化trie树及其应用
可持久化trie树 可持久化trie树现在想来是比较好理解的了,但却看了一个下午... 相当于对于每个状态建立一条链(或者说一棵trie),求解的时候只要让两个点按照相同的步子走然后看sum的大小关系 ...
- 可持久化Trie树初步
可持久化Trie树和可持久化线段树很像,依次插入信息,通过减法来进行历史版本查询. 2015年11月27日 bzoj3261 最大异或和 我们需要计算 a[p] xor a[p+1] xor ... ...
- [十二省联考2019]异或粽子——可持久化trie树+堆
题目链接: [十二省联考2019]异或粽子 求前$k$大异或区间,可以发现$k$比较小,我们考虑找出每个区间. 为了快速得到一个区间的异或和,将原序列做前缀异或和. 对于每个点作为右端点时,我们维护出 ...
- BZOJ4477[Jsoi2015]字符串树——可持久化trie树
题目描述 萌萌买了一颗字符串树的种子,春天种下去以后夏天就能长出一棵很大的字符串树.字符串树很奇特,树枝上都密密麻麻写满了字符串,看上去很复杂的样子.[问题描述]字符串树本质上还是一棵树,即N个节点N ...
- BZOJ5338 [TJOI2018] Xor 【可持久化Trie树】【dfs序】
题目分析: 很无聊的一道题目.首先区间内单点对应异或值的询问容易想到trie树.由于题目在树上进行,case1将路径分成两段,然后dfs的时候顺便可持久化trie树做询问.case2维护dfs序,对d ...
- 51nod 1295 XOR key (可持久化Trie树)
1295 XOR key 题目来源: HackerRank 基准时间限制:1.5 秒 空间限制:262144 KB 分值: 160 难度:6级算法题 给出一个长度为N的正整数数组A,再给出Q个查 ...
随机推荐
- yarn架构——本质上是在做解耦 将资源分配和应用程序状态监控两个功能职责分离为RM和AM
Hadoop YARN架构解读 原Mapreduce架构 原理架构图如下: 图 1.Hadoop 原 MapReduce 架构 原 MapReduce 程序的流程:首先用户程序 (JobClient) ...
- Memched——C#操作
Memched还是比较简单的,这里把C#的相关操作整理了一下,Mark~ /// <summary> /// 缓存操作类. /// </summary> /// <rem ...
- eclipse搭建android开发环境
1.首先安装JDK 此步骤是做JAVA必经之路,不多累述,强调要注意的地方: 目前为止android的开发环境只支持JDK1.7,千万不要下载JDK1.8. 下载的JDK一定要选择好操作系统,特别是要 ...
- Servlet基础(二)
1.什么是Servlet java类,提供web形式的访问 servlet就是按照javaee中servlet规范所编写的java类 能够被浏览器通过URL形式访问到 2.怎么在javae ...
- vue中的事件修饰符
vue提倡的是在方法中只有对数据的处理,所以提供了事件修饰符用于DOM的事件处理,常用的事件修饰符有以下几个: (1). stop:阻止冒泡(通俗讲就是阻止事件向上级DOM元素传递) 点击内层div的 ...
- DirectUI界面编程(三)从XML文件中加载界面
Duilib支持xml界面布局,使得界面设计与逻辑处理相分离,本节介绍如何从xml文件中加载界面元素. 我们需要以下几个步骤: 创建并初始化CPaintManagerUI对象. 创建CDialogBu ...
- 错误:com.android.builder.packaging.DuplicateFileException: Duplicate files copied
File2: C:\Users\guoxw\.gradle\caches\modules-2\files-2.1\org.jsoup\jsoup\1.10.3\b842f960942503cf1abb ...
- Xml实现图片旋转
1. 需求:不使用Java代码,实现旋转图片动画 2.实现:使用Progressbar控件 3. anim/anim_loading.xml <?xml version="1.0&qu ...
- 子线程创建AlertDialog错误
Can't create handler inside thread that has not called Looper.prepare()
- Vue中的坑
今天给大家总结了Vue中的坑,都是平常踩到的,希望对大家有所帮助 1.vuex的一个“坑” 报错原因:在使用Vuex的时候在main.js中引入的时候对名字的大小写有区别 解决法案: 错误写法: 正确 ...