【bzoj4260】Codechef REBXOR Trie树

题目描述

输入

输入数据的第一行包含一个整数N,表示数组中的元素个数。

第二行包含N个整数A1,A2,…,AN。

输出

输出一行包含给定表达式可能的最大值。

样例输入

5
1 2 3 1 2

样例输出

6



题解

Trie树

还是本着100000(2)>011111(2)的原则,显然需要从高位贪心取值。

又因为x^x=0,所以有$a_l\oplus a_{l+1}\oplus \cdots\oplus a_{r-1}\oplus a_r=(a_1\oplus a_2\oplus\cdots\oplus a_{l-2}\oplus a_{l-1})\oplus(a_1\oplus a_2\oplus\cdots\oplus a_{r-1}\oplus a_{r})$

这样如果我们知道了r,就能够贪心求出最大的异或和。

所以我们可以从左往右将前缀异或和按位加入到Trie树中,并在每次添加后贪心求出最大的异或和,记为f[i],表示以i结尾的连续子序列的异或和的最大值。

然后再维护f[i]的前缀最大值,为1~i中所有连续子序列的异或和的最大值。

从右往左进行同样的操作,为i~n中所有连续子序列的异或和的最大值。

这样,我们枚举一个位置,它左边的最大值和它右边的最大值之和就是所求,求一下最大值即为答案。

#include <cstdio>
#include <algorithm>
#define N 400010
using namespace std;
struct data
{
	int next[N * 30][2] , tot;
	void insert(int v)
	{
		int x = 0 , i;
		bool t;
		for(i = 1 << 30 ; i ; i >>= 1)
		{
			t = v & i;
			if(!next[x][t]) next[x][t] = ++tot;
			x = next[x][t];
		}
	}
	int query(int v)
	{
		int x = 0 , i , ret = 0;
		bool t;
		for(i = 1 << 30 ; i ; i >>= 1)
		{
			t = v & i;
			if(next[x][t ^ 1]) ret += i , x = next[x][t ^ 1];
			else x = next[x][t];
		}
		return ret;
	}
}A , B;
int a[N] , m1[N] , m2[N];
int main()
{
	int n , i , t , ans = 0;
	scanf("%d" , &n);
	for(i = 1 ; i <= n ; i ++ ) scanf("%d" , &a[i]);
	A.insert(0) , B.insert(0);
	for(t = 0 , i = 1 ; i <= n ; i ++ ) t ^= a[i] , A.insert(t) , m1[i] = max(m1[i - 1] , A.query(t));
	for(t = 0 , i = n ; i >= 1 ; i -- ) t ^= a[i] , B.insert(t) , m2[i] = max(m2[i + 1] , B.query(t));
	for(i = 1 ; i < n ; i ++ ) ans = max(ans , m1[i] + m2[i + 1]);
	printf("%d\n" , ans);
	return 0;
}
时间: 2024-11-12 04:31:01

【bzoj4260】Codechef REBXOR Trie树的相关文章

bzoj4260: Codechef REBXOR

求异或maxmin一般用trie (二进制式的trie).query中找的是满足((x>>i)&1)^A=1,那么A=((x>>i)&1)^1:maxx=max(sumx,sumi)(i=[1,x]).(YY一下异或的性质 #include<cstdio> #include<cstring> #include<iostream> #include<algorithm> using namespace std; #def

BZOJ 4260: Codechef REBXOR( trie )

求出前缀和, 那么以第x个元素结尾的最大异或值是max(sumx^sump)(1≤p<x), 用trie加速. 后缀同理, 然后扫一遍就OK了.时间复杂度O(31N) ----------------------------------------------------------------------- #include<cstdio> #include<cstring> #include<algorithm> #include<cctype>

BZOJ4260 Codechef REBXOR 题解

题目大意: 给定一个长度为n的序列,求1≤l1≤r1<l2≤r2≤n使得(⊕r1i=l1ai)+(⊕r2i=l2ai)最大,输出这个最大值. 思路: 用Trie求出前缀异或和以及后缀异或和,再求出前缀异或和以及后缀异或和中最大的,前后相加,求最大值.用可持久化Trie求异或和也可. 代码: Trie 1 #include<cstdio> 2 #include<cstring> 3 #include<iostream> 4 #define M 400009 5 us

【BZOJ4260】 Codechef REBXOR 可持久化Trie

看到异或就去想前缀和(⊙o⊙) 这个就是正反做一遍最大异或和更新答案 最大异或就是很经典的可持久化Trie,从高到低贪心 WA: val&(1<<(base-1))得到的并不直接是1/0 1 #include<bits/stdc++.h> 2 #define ll long long 3 #define N 400005 4 using namespace std; 5 inline int read(){ 6 int x=0,f=1;char ch=getchar();

【trie树专题】

[map || trie]P2580 于是他错误的点名开始了 题目背景 XS中学化学竞赛组教练是一个酷爱炉石的人. 他会一边搓炉石一边点名以至于有一天他连续点到了某个同学两次,然后正好被路过的校长发现了然后就是一顿欧拉欧拉欧拉(详情请见已结束比赛CON900). 题目描述 这之后校长任命你为特派探员,每天记录他的点名.校长会提供化学竞赛学生的人数和名单,而你需要告诉校长他有没有点错名.(为什么不直接不让他玩炉石.) 输入格式 第一行一个整数 n,表示班上人数.接下来 n 行,每行一个字符串表示其

poj3630 Phone List (trie树模板题)

Phone List Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 26328   Accepted: 7938 Description Given a list of phone numbers, determine if it is consistent in the sense that no number is the prefix of another. Let's say the phone catalogu

跳跃表,字典树(单词查找树,Trie树),后缀树,KMP算法,AC 自动机相关算法原理详细汇总

第一部分:跳跃表 本文将总结一种数据结构:跳跃表.前半部分跳跃表性质和操作的介绍直接摘自<让算法的效率跳起来--浅谈"跳跃表"的相关操作及其应用>上海市华东师范大学第二附属中学 魏冉.之后将附上跳跃表的源代码,以及本人对其的了解.难免有错误之处,希望指正,共同进步.谢谢. 跳跃表(Skip List)是1987年才诞生的一种崭新的数据结构,它在进行查找.插入.删除等操作时的期望时间复杂度均为O(logn),有着近乎替代平衡树的本领.而且最重要的一点,就是它的编程复杂度较同类

Trie树学习2

数组实现的Trie树 字符容量有限,可以使用链表实现更为大容量的Trie #include <iostream> #include <cstdio> #include <string> #include <cstring> #include <vector> #include <map> #include <set> #include <algorithm> #include <cstdlib> #

trie树(字典树)

1. trie树,又名字典树,顾名思义,它是可以用来作字符串查找的数据结构,它的查找效率比散列表还要高. trie树的建树: 比如有字符串"ab" ,"adb","adc"   可以建立字典树如图: 树的根节点head不存储信息,它有26个next指针,分别对应着字符a,b,c等.插入字符串ab时,next['a'-'a']即next[0]为空,这是申请一个结点放在next[0]的位置,插入字符串db时,next['d'-'a']即next[3]