http://www.lydsy.com/JudgeOnline/problem.php?id=2741

分块或可持久化trie

可以先看看这个:高斯消元解XOR方程组

分块做法:

我们先求出前i个数的异或和,即b[i]=a[1]^a[2]^...^a[i],不失一般性,记b[0]=0。

那么a[i]^a[i+1]^...^a[j-1]^a[j]=b[j]^b[i-1]。

所以原问题变成在b[l-1...r]中任选2个数,使得异或和最大。

我们将0..N分成$\sqrt{N}$块,不妨记第i块的左端为l[i],右端为r[i]。

每个块建一棵二进制树,将所有的数插入到所在的块的二进制树中。

然后记$f[i][j]$表示b[i]与第j块中某一个数异或(即b[i]与b[l[j]...r[j]]中的某一个数异或),得到的最大的异或和是多少。这个可以在$O(31N\sqrt{N})$的时间内解决。

$g[i][j]$表表示第i块中某一个数与第j块中某一个数异或(即b[l[i]...r[i]]中的某一个数异或与b[l[j]...r[j]]中的某一个数异或),得到的最大的异或和是多少。这个可以在$O(N\sqrt{N})$从F数组得到。

现在预处理已经完成了。

对于询问区间[l-1...r],分类讨论一下就可以了,时间复杂度是$O(MN)$

所以总的时间复杂度是$O(31N\sqrt{N}+MN)$

#include<cstdio>
#include<cstdlib>
#include<iostream>
#include<fstream>
#include<algorithm>
#include<cstring>
#include<string>
#include<cmath>
#include<queue>
#include<stack>
#include<map>
#include<utility>
#include<set>
#include<bitset>
#include<vector>
#include<functional>
#include<deque>
#include<cctype>
#include<climits>
#include<complex>
//#include<bits/stdc++.h>适用于CF,UOJ,但不适用于poj using namespace std; typedef long long LL;
typedef double DB;
typedef pair<int,int> PII;
typedef complex<DB> CP; #define mmst(a,v) memset(a,v,sizeof(a))
#define mmcy(a,b) memcpy(a,b,sizeof(a))
#define fill(a,l,r,v) fill(a+l,a+r+1,v)
#define re(i,a,b) for(i=(a);i<=(b);i++)
#define red(i,a,b) for(i=(a);i>=(b);i--)
#define ire(i,x) for(typedef(x.begin()) i=x.begin();i!=x.end();i++)
#define fi first
#define se second
#define m_p(a,b) make_pair(a,b)
#define p_b(a) push_back(a)
#define SF scanf
#define PF printf
#define two(k) (1<<(k)) template<class T>inline T sqr(T x){return x*x;}
template<class T>inline void upmin(T &t,T tmp){if(t>tmp)t=tmp;}
template<class T>inline void upmax(T &t,T tmp){if(t<tmp)t=tmp;} const DB EPS=1e-;
inline int sgn(DB x){if(abs(x)<EPS)return ;return(x>)?:-;}
const DB Pi=acos(-1.0); inline int gint()
{
int res=;bool neg=;char z;
for(z=getchar();z!=EOF && z!='-' && !isdigit(z);z=getchar());
if(z==EOF)return ;
if(z=='-'){neg=;z=getchar();}
for(;z!=EOF && isdigit(z);res=res*+z-'',z=getchar());
return (neg)?-res:res;
}
inline LL gll()
{
LL res=;bool neg=;char z;
for(z=getchar();z!=EOF && z!='-' && !isdigit(z);z=getchar());
if(z==EOF)return ;
if(z=='-'){neg=;z=getchar();}
for(;z!=EOF && isdigit(z);res=res*+z-'',z=getchar());
return (neg)?-res:res;
} const int maxN=;
const int maxcnt=; struct Tnode
{
Tnode *son[];
inline Tnode(){son[]=son[]=;}
}; int N,Q;
int len,cnt,id[maxN+],l[maxcnt+],r[maxcnt+];
int a[maxN+];
Tnode *rt[maxcnt+];
int f[maxN+][maxcnt+];
int g[maxcnt+][maxcnt+];
int lastans; int t[];
inline void insert(Tnode *p,int v)
{
int i;
re(i,,)t[i]=v&,v>>=;
red(i,,)
{
if(!p->son[t[i]])p->son[t[i]]=new Tnode;
p=p->son[t[i]];
}
}
inline int find(Tnode *p,int v)
{
int i,res=;
re(i,,)t[i]=v&,v>>=;
red(i,,)
{
int to=(p->son[t[i]^])?t[i]^:t[i];
res=(res<<)+to;
p=p->son[to];
}
return res;
} int main()
{
freopen("bzoj2741.in","r",stdin);
freopen("bzoj2741.out","w",stdout);
int i,j;
N=gint()+;Q=gint();
re(i,,N)a[i]=gint();
re(i,,N)a[i]^=a[i-];
len=int(sqrt(DB(N)));
re(i,,N)
{
if((i-)%len==)r[cnt]=i-,l[++cnt]=i;
id[i]=cnt;
}
r[cnt]=N;
re(i,,cnt)rt[i]=new Tnode;
re(i,,N)insert(rt[id[i]],a[i]);
re(i,,N)re(j,,cnt)f[i][j]=a[i]^find(rt[j],a[i]);
re(i,,N)re(j,,cnt)upmax(g[id[i]][j],f[i][j]);
lastans=;
while(Q--)
{
int x=gint(),y=gint();
int L=int( min( (LL(x)+LL(lastans))%LL(N-)+ , (LL(y)+LL(lastans))%LL(N-)+ ) ) + ;
int R=int( max( (LL(x)+LL(lastans))%LL(N-)+ , (LL(y)+LL(lastans))%LL(N-)+ ) ) + ;
L--;
lastans=;
if(id[L]==id[R] || id[L]+==id[R])
{
re(i,L,R)re(j,i,R)upmax(lastans,a[i]^a[j]);
}
else
{
int p=(L==l[id[L]])?id[L]:id[L]+,q=(R==r[id[R]])?id[R]:id[R]-;
re(i,p,q)re(j,i,q) upmax(lastans,g[i][j]);
re(i,L,l[p]-)re(j,p,q) upmax(lastans,f[i][j]);
re(i,r[q]+,R)re(j,p,q) upmax(lastans,f[i][j]);
re(i,L,l[p]-)re(j,r[q]+,R)upmax(lastans,a[i]^a[j]);
re(i,L,l[p]-)re(j,i+,l[p]-)upmax(lastans,a[i]^a[j]);
re(i,r[q]+,R)re(j,i+,R)upmax(lastans,a[i]^a[j]);
}
cout<<lastans<<endl;
}
return ;
}

可持久化trie做法:

可持久化trie的想法比较简单。

第i棵trie中插入了b[0...i],我们可以从第i-1棵trie得到。

对于询问区间[l-1...r],枚举其中一个数,然后同时在第r棵trie和第l-2棵trie中走即可。

总的时间复杂度是$O(31MN)$

bzoj2741【FOTILE模拟赛】L的更多相关文章

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

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

  2. BZOJ2741:[FOTILE模拟赛]L

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

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

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

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

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

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

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

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

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

  7. bzoj 2741 [FOTILE模拟赛] L

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

  8. 【BZOJ2741】【FOTILE模拟赛】L 分块+可持久化Trie树

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

  9. BZOJ2741: 【FOTILE模拟赛】L

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

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

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

随机推荐

  1. Guice学习(一)

    Guice学习(一) Guice是Google开发的一个轻量级依赖注入框架(IOC).Guice非常小而且快,功能类似与Spring,但效率上网上文档显示是它的100倍,而且还提供对Servlet,A ...

  2. C#量转换为汉字表达

    /* 创造者:菜刀打好博客  * 创建日期: 2014年09一个月04号码  * 特征:Money类型转换  *  */ namespace Net.String.ConsoleApplication ...

  3. swift实现ios类似微信输入框跟随键盘弹出的效果

    封面(图文无关) 为什么要做这个效果 在聊天app,例如微信中,你会注意到一个效果,就是在你点击输入框时输入框会跟随键盘一起向上弹出,当你点击其他地方时,输入框又会跟随键盘一起向下收回,二者完全无缝连 ...

  4. Linux下查看所有用户(shell脚本获取)

      在Linux系统中,使用者账号管理最重要的两个文件是/etc/password和/etc/shadow.在/etc/password文件中,每一行都代表一个账号,但是有很多账号是系统账号.比如:b ...

  5. 复杂 Listview 显示 多个样式

    三种方式 目前为止有三种方法让Listview现实多个样式 最简单最常用的,通过addHeaderView或addFooterView,但是只能在首尾添加 较麻烦但正规的方式,通过getViewTyp ...

  6. Ajax调用webService(一) 不跨域。

    注:需要的js文件与组件(jquery-1.4.2.min.js和Newtonsoft.Json) 同域:要调用的webservice与ajax请求页面在同一个网站下(本文中都是本地测试). 数据库( ...

  7. php学习之路

    1.php拼接字符串+查询 $floor_id = M('house_floor_input')->where($map1)->field('id')->select(); $flo ...

  8. C# 截取字符串,区分中英文情况

    public static string SubstringByCN(string strInput, int begin, int length) { if (!Regex.IsMatch(strI ...

  9. IO流文件字符输入输出流,缓冲流

    由于字节输入输出流在操纵Unicode字符时可能有乱码现象 于是就有了操作字符的输入输出流 Reader ,Writer和他们的子类FileReader,FileWrite(其实就是用来辅助构造的 W ...

  10. jquery中的replaceWith()和html()有什么区别?

    区别在于,html()会替换指定元素内部的HTML,而replaceWith()会替换元素本身及其内部的HTML. 例子: 1 <div id="myid" /> 1 ...