传送门

解题思路

  首先求出前缀异或和,那么问题就转化成了区间内选两个数使得其异或和最大。数据范围不是很大考虑分块,设\(f[x][i]\)表示第\(x\)块开头到\(i\)这个位置与\(a[i]\)异或得到的最大的数,而对\(f\)求前缀\(max\)就可以得出每一块的开头到后面任意一点的区间内异或最大。而求\(f\)的过程实际是从区间内取一个数和给定数异或和最大,那么这个可以用\(0/1\) \(Trie\)来做,就可以造一棵可持久化\(Trie\)。询问时整块直接调用\(f\),前面的小块直接用可持久化\(Trie\)求解,时间复杂度\(O(n\sqrt(n) logn)\)

代码

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath> using namespace std;
const int N=12005;
const int M=N*60;
typedef long long LL; template<class T> void rd(T &x){
x=0; char ch=getchar();
while(!isdigit(ch)) ch=getchar();
while(isdigit(ch)) x=(x<<1)+(x<<3)+ch-'0',ch=getchar();
}
inline int max(int x,int y) {return x>y?x:y;} int n,m,siz,num,rt[N],tot,bl[N],l[N],r[N];
int a[N],f[205][N],lstans,zz[N]; struct Trie{
int ch[M][2],sum[M];
void insert(int pre,int &x,int now,int d){
x=++tot; sum[x]=sum[pre]+1; if(!d) return ;
ch[x][0]=ch[pre][0]; ch[x][1]=ch[pre][1];
if(now&(1<<(d-1))) insert(ch[pre][1],ch[x][1],now,d-1);
else insert(ch[pre][0],ch[x][0],now,d-1);
}
int query(int u,int v,int now,int d){
if(!d) return 0;
if(now&(1<<(d-1))) {
if(sum[ch[u][0]]-sum[ch[v][0]]!=0)
return (query(ch[u][0],ch[v][0],now,d-1))|(1<<(d-1));
else return query(ch[u][1],ch[v][1],now,d-1);
}
else {
if(sum[ch[u][1]]-sum[ch[v][1]]!=0)
return (query(ch[u][1],ch[v][1],now,d-1)|(1<<(d-1)));
else return query(ch[u][0],ch[v][0],now,d-1);
}
}
}tree; inline void prework(){
for(int i=1;i<=num;++i)
for(int j=l[i]+1;j<=n;++j)
f[i][j]=tree.query(rt[j],rt[l[i]-1],a[j],31);
// for(int i=1;i<=num;i++)
// for(int j=l[i]+1;j<=n;j++)
// printf("f[%d][%d]=%lld\n",i,j,f[i][j]);
for(int i=1;i<=num;++i)
for(int j=l[i]+1;j<=n;++j)
f[i][j]=max(f[i][j],f[i][j-1]);
} inline int ask(int x,int y){
int ret=0;
if(bl[x]==bl[y]){
for(int i=x;i<y;++i)
ret=max(ret,tree.query(rt[y],rt[max(0,x-1)],a[i],31));
return ret;
}
ret=f[bl[x]+1][y];
for(int i=x;i<=r[bl[x]];++i)
ret=max(ret,tree.query(rt[y],rt[max(0,x-1)],a[i],31));
return ret;
} signed main(){
rd(n),rd(m); siz=sqrt(n);
num=n/siz; if(n%siz) num++;
for(int i=1;i<=n;++i){
rd(a[i]); bl[i]=(i-1)/siz+1; a[i]^=a[i-1];
tree.insert(rt[i-1],rt[i],a[i],31);
}
for(int i=1;i<=num;++i) l[i]=(i-1)*siz+1,r[i]=i*siz;
r[num]=n; prework(); int L,R;
while(m--){
rd(L); rd(R);
L=((LL)lstans+L)%n+1,R=((LL)lstans+R)%n+1;
if(L>R) swap(L,R); lstans=ask(L-1,R);
printf("%d\n",lstans);
}
return 0;
}

BZOJ 2741: 【FOTILE模拟赛】L(可持久化Trie+分块)的更多相关文章

  1. BZOJ.2741.[FOTILE模拟赛]L(分块 可持久化Trie)

    题目链接 首先记\(sum\)为前缀异或和,那么区间\(s[l,r]=sum[l-1]^{\wedge}sum[r]\).即一个区间异或和可以转为求两个数的异或和. 那么对\([l,r]\)的询问即求 ...

  2. bzoj 2741 [FOTILE模拟赛] L

    Description 多个询问l,r,求所有子区间异或和中最大是多少 强制在线 Solution 分块+可持久化trie 1.对于每块的左端点L,预处理出L到任意一个i,[L,j] 间所有子区间异或 ...

  3. 【bzoj2741】[FOTILE模拟赛]L 可持久化Trie树+分块

    题目描述 FOTILE得到了一个长为N的序列A,为了拯救地球,他希望知道某些区间内的最大的连续XOR和. 即对于一个询问,你需要求出max(Ai xor Ai+1 xor Ai+2 ... xor A ...

  4. 【BZOJ2741】【块状链表+可持久化trie】FOTILE模拟赛L

    Description FOTILE得到了一个长为N的序列A,为了拯救地球,他希望知道某些区间内的最大的连续XOR和. 即对于一个询问,你需要求出max(Ai xor Ai+1 xor Ai+2 .. ...

  5. BZOJ2741 FOTILE模拟赛L(分块+可持久化trie)

    显然做个前缀和之后变成询问区间内两个数异或最大值. 一种暴力做法是建好可持久化trie后直接枚举其中一个数查询,复杂度O(nmlogv). 观察到数据范围很微妙.考虑瞎分块. 设f[i][j]为第i个 ...

  6. 【bzoj2741】[FOTILE模拟赛] L

    Portal --> bzoj2741 Solution 突然沉迷分块不能自拔 考虑用分块+可持久化trie来解决这个问题 对于每一块的块头\(L\),预处理\([L,i]\)区间内的所有子区间 ...

  7. BZOJ2741:[FOTILE模拟赛]L

    Description FOTILE得到了一个长为N的序列A,为了拯救地球,他希望知道某些区间内的最大的连续XOR和. 即对于一个询问,你需要求出max(Ai xor Ai+1 xor Ai+2 .. ...

  8. bzoj 2741: 【FOTILE模拟赛】L 分塊+可持久化trie

    2741: [FOTILE模拟赛]L Time Limit: 15 Sec  Memory Limit: 162 MBSubmit: 1116  Solved: 292[Submit][Status] ...

  9. 【BZOJ】【2741】【FOTILE模拟赛】L

    可持久化Trie+分块 神题……Orz zyf & lyd 首先我们先将整个序列搞个前缀异或和,那么某一段的异或和,就变成了两个数的异或和,所以我们就将询问[某个区间中最大的区间异或和]改变成 ...

随机推荐

  1. MongoDB分片配置 优化 不错

    简单注解:mongos 路由进程, 应用程序接入mongos再查询到具体分片,监听端口默认27017config server 路由表服务, 每一台都具有全部chunk的路由信息 shard为数据存储 ...

  2. C语言readdir()函数:读取目录函数

    相关函数:open, opendir, closedir, rewinddir, seekdir, telldir, scandir 头文件:#include <sys/types.h> ...

  3. Recurrent Neural Network(2):BPTT and Long-term Dependencies

    在RNN(1)中,我们将带有Reccurent Connection的node依照时间维度展开成了如下的形式: 在每个时刻t=0,1,2,3,...,神经网络的输出都会产生error:E0,E1,E2 ...

  4. Hadoop: 在Azure Cluster上使用MapReduce

    Azure对于学生账户有260刀的免费试用,火急火燎地创建Hadoop Cluster!本例子是使用Hadoop MapReduce来统计一本电子书中各个单词的出现个数. Let's get hand ...

  5. Java基础复习(1)

    1. Java 基本数据类型 参考博客: https://www.cnblogs.com/LiaHon/p/11043238.html Java语言提供了八种基本类型. 六种数字类型(四个整数型,两个 ...

  6. HTML DOM cursor 属性

    值 描述 url 需被使用的自定义光标的URL 注释:请在此列表的末端始终定义一种普通的光标,以防没有由 URL 定义的可用光标. default 默认光标(通常是一个箭头) auto 默认.浏览器设 ...

  7. nmon内存分析

    可参考: MemTotal:显示当前服务器物理内存大小,本服务器有8063180 KB≍7874 MB左右. MemFree:显示当前服务器的空闲内存大小,本服务器有5052336 KB≍4934 M ...

  8. 矩阵快速幂(queue递推)

    http://acm.hdu.edu.cn/showproblem.php?pid=2604 Queuing Time Limit: 10000/5000 MS (Java/Others)    Me ...

  9. Meet in the middle算法总结 (附模板及SPOJ ABCDEF、BZOJ4800、POJ 1186、BZOJ 2679 题解)

    目录 Meet in the Middle 总结 1.算法模型 1.1 Meet in the Middle算法的适用范围 1.2Meet in the Middle的基本思想 1.3Meet in ...

  10. yum安装时出现No more mirrors to try.

    可能原因:可能是不正常删除造成的 解决方法: yum clean allyum makecacheyum -y update 然后重新安装