bzoj 2741 可持久化trie
首先我们设si为前i个数的xor和,那么对于询问区间[i,j]的xor和,就相当于si-1^sj,那么对于这道题的询问我们可以处理处si,然后对于询问[l,r],可以表示为在区间[l-1,r]里找两个数使得这两个数的xor值最大,对于区间中找一个数使得xor一个已知数的值最大我们可以用可持久化trie来完成(有疑问请移步http://www.cnblogs.com/BLADEVIL/p/3669219.html),设询问query(l,r,x)表示x与区间[l,r]的数的xor值最大值。那么我们可以将这n个数分为sqrt(n)个块,我们设head[i]为第i个块的最开始元素,w[i][j]为从第i个块的第一个元素开始,到第j个数的区间中,任意找两个数的xor值最大,那么易得到转移方程w[i][j]=max(w[i][j],w[i][j-1]+query(head[i],j-1,a[j]))。这个的时间复杂度是o(sqrt(n)*n*logn)的,然后对于询问[l,r],我们设l后面第一个块为x块,那么ans=max(w[x][r],max(l,r,a[j])),j∈[l,head[x]-1]。
备注:开始的时候以为题目说a[i]不会超过32767,后来发现好像是2147483647,然后改了trie的深度之后发现总是RE,因为我写的trie是先建一个空树,然后再插入节点的,31层的满二叉树当然没有办法建出来,那么改成了直接插入,发现还是WA,因为上一次的答案可能只比2147483647小一点,那么加上这次询问的值就会超过int,所以先强制转成long long然后%n就好了。
开始以为RE是内存的问题,所以开大了好多,懒得改了就这样吧。
/**************************************************************
Problem: 2741
User: BLADEVIL
Language: C++
Result: Accepted
Time:8148 ms
Memory:41680 kb
****************************************************************/
//By BLADEVIL
#include <cmath>
#include <cstdio>
#include <cstring>
#include <algorithm>
#define maxn 20000
#define maxm 400
using namespace std;
struct tree {
int son[];
int cnt;
tree() {
memset(son,,sizeof son);
cnt=;
}
}t[*maxn];
struct rec {
int key,num,rot;
rec() {
key=num=rot=;
}
}a[maxn];
int n,m,len,sum,tot;
int w[maxm][maxn],head[maxm];
void insert(int &x,int rot,int y,int dep) {
if (!x) x=++tot;
if (dep==-) {
t[x].cnt=t[rot].cnt+;
return ;
}
if (y&(<<dep)) {
insert(t[x].son[],t[rot].son[],y,dep-);
t[x].son[]=t[rot].son[];
} else {
insert(t[x].son[],t[rot].son[],y,dep-);
t[x].son[]=t[rot].son[];
}
t[x].cnt=t[rot].cnt+;
}
int query(int lx,int rx,int y,int dep) {
if (dep==-) return ;
if (y&(<<dep)) {
if (t[t[rx].son[]].cnt-t[t[lx].son[]].cnt)
return (<<dep)+query(t[lx].son[],t[rx].son[],y,dep-); else
return query(t[lx].son[],t[rx].son[],y,dep-);
} else {
if (t[t[rx].son[]].cnt-t[t[lx].son[]].cnt)
return (<<dep)+query(t[lx].son[],t[rx].son[],y,dep-); else
return query(t[lx].son[],t[rx].son[],y,dep-);
}
}
int main() {
scanf("%d%d",&n,&m);
for (int i=;i<=n;i++) scanf("%d",&a[i].key);
for (int i=;i<=n;i++) a[i].key^=a[i-].key;
sum=len=sqrt(n); if (len*len<n) len++,sum=(n+len-)/len;
//printf("%d %d\n",len,sum);
for (int i=;i<=n;i++) if (!head[a[i].num=(i+len-)/len]) head[a[i].num]=i; head[sum+]=n+;
//for (int i=1;i<=n;i++) printf("%d ",a[i].key); printf("\n");
//for (int i=1;i<=sum;i++) printf("%d ",head[i]);
for (int i=;i<=n;i++) insert(a[i].rot,a[i-].rot,a[i].key,);
for (int i=;i<=sum;i++)
for (int j=head[i];j<=n;j++)
w[i][j]=max(w[i][j-],query(a[head[i]-].rot,a[j].rot,a[j].key,));
/*
for (int i=1;i<=sum;i++) {
for (int j=1;j<=n;j++) printf("%d ",w[i][j]);
printf("\n");
}
*/
int ans=;
while (m--) {
int l,r; scanf("%d%d",&l,&r); int x,y;
x=((long long)l+ans)%n+; y=((long long)r+ans)%n+;
l=min(x,y); r=max(x,y); ans=; l--;
if (a[l].num==a[r].num) {
for (int i=l;i<=r;i++) ans=max(ans,query(a[l-].rot,a[r].rot,a[i].key,));
} else {
ans=w[a[l].num+][r]; //printf("%d %d %d\n",ans,a[l].num+1,r);
for (int i=l;i<=head[a[l].num+]-;i++) ans=max(ans,query(a[l-].rot,a[r].rot,a[i].key,));
}
printf("%d\n",ans);
}
return ;
}
bzoj 2741 可持久化trie的更多相关文章
- BZOJ - 3166 可持久化Trie 维护次大区间
题意:给出\(a[1...n]\),找出一个连续区间\(a[l...r],r>l\),令该区间的次大值为\(a_k\),使得\(a_k⊕a_i,l≤i≤r\)最大,输出全局最优解 (这题意有点别 ...
- [BZOJ3261&BZOJ3166]可持久化trie树及其应用
可持久化trie树 可持久化trie树现在想来是比较好理解的了,但却看了一个下午... 相当于对于每个状态建立一条链(或者说一棵trie),求解的时候只要让两个点按照相同的步子走然后看sum的大小关系 ...
- bzoj 2741: 【FOTILE模拟赛】L 分塊+可持久化trie
2741: [FOTILE模拟赛]L Time Limit: 15 Sec Memory Limit: 162 MBSubmit: 1116 Solved: 292[Submit][Status] ...
- BZOJ 2741 【FOTILE模拟赛】L(可持久化trie)
http://www.lydsy.com/JudgeOnline/problem.php?id=2741 思路:我们先将a变成a的异或前缀,这样问题就变成了,在l-1到r区间内,找出i,j令a[i]^ ...
- bzoj 2741 分块+可持久化trie
多个询问l,r,求所有子区间异或和中最大是多少 强制在线 做法: 分块+可持久化trie 1.对于每块的左端点i,预处理出i到任意一个j,()i,j)间所有子区间异或和中最大为多少,复杂度O(\(n\ ...
- BZOJ 2741: 【FOTILE模拟赛】L [分块 可持久化Trie]
题意: 区间内最大连续异或和 5点调试到现在....人生无望 但总算A掉了 一开始想错可持久化trie的作用了...可持久化trie可以求一个数与一个数集(区间中的一个数)的最大异或和 做法比较明显, ...
- BZOJ.2741.[FOTILE模拟赛]L(分块 可持久化Trie)
题目链接 首先记\(sum\)为前缀异或和,那么区间\(s[l,r]=sum[l-1]^{\wedge}sum[r]\).即一个区间异或和可以转为求两个数的异或和. 那么对\([l,r]\)的询问即求 ...
- BZOJ 2741: 【FOTILE模拟赛】L(可持久化Trie+分块)
传送门 解题思路 首先求出前缀异或和,那么问题就转化成了区间内选两个数使得其异或和最大.数据范围不是很大考虑分块,设\(f[x][i]\)表示第\(x\)块开头到\(i\)这个位置与\(a[i]\)异 ...
- [BZOJ 4103] [Thu Summer Camp 2015] 异或运算 【可持久化Trie】
题目链接:BZOJ - 4103 题目分析 THUSC滚粗之后一直没有写这道题,从来没写过可持久化Trie,发现其实和可持久化线段树都是一样的.嗯,有些东西就是明白得太晚. 首先Orz ZYF-ZYF ...
随机推荐
- CentOS6.5 重启网络报错:Bringing up interface eth0: Error: Connection activation failed: Device not managed by NetworkManager or unavailable
CentOS6.5 重启网络报错: Bringing up interface eth0: Error: Connection activation failed: Device not manage ...
- ROC曲线【转】
受试者工作特征曲线(receiver operating characteristic curve, 简称ROC曲线),又称为感受性曲线(sensitivity curve).得此名的原因在于曲线上各 ...
- bzoj1050[HAOI2006]旅行comf(枚举+贪心+并查集)
Description 给你一个无向图,N(N<=500)个顶点, M(M<=5000)条边,每条边有一个权值Vi(Vi<30000).给你两个顶点S和T,求一条路径,使得路径上最大 ...
- 用Racket语言写了一个万花筒的程序
用Racket语言写了一个万花筒的程序 来源:https://blog.csdn.net/chinazhangyong/article/details/79362394 https://github. ...
- Qt浅谈之总结(整理)
Qt浅谈之总结(整理) 来源 http://blog.csdn.net/taiyang1987912/article/details/32713781 一.简介 QT的一些知识点总结,方便以后查阅. ...
- hbase 基本的JavaApi 数据操作及数据过滤(filter)
本文主要是hbase的表操作.数据操作.数据查询过滤等,如果对JDBC或ADO有了解,容易理解HBASE API. hbase版本是2.0. 1.为了方便先贴helper的部分代码(文末git上有完整 ...
- Gradle及eclipse插件安装
1. 下载Gradle Gradle需要jdk7或以上版本,使用Java –version命令进行测试:Gradle自带了Groovy库,所以无需再安装Groovy,已经安装的Groovy会被Gra ...
- C中 ->运算符说明
岁数大了,记忆力不好!这里记下,以后忘了来查! ->运算符. 访问结构中的成员 用 点“.”运算符 Ex: typedef struct st { char a; int b; } st; 定义 ...
- 解题:POI 2015 Pieczęć
题面 发现好像没有什么好做法,那就模拟么=.= 以印章左上角的'x'为基准,记录印章上'x'的相对位置模拟.记录相对位置是因为可能有这种情况↓ 直接模拟是会漏掉的=.= #include<cst ...
- mysql四-2:多表查询
一 介绍 本节主题 多表连接查询 复合条件连接查询 子查询 准备表 #建表 create table department( id int, name varchar(20) ); create ta ...