• [1525] Cow Xor

  • 时间限制: 2000 ms 内存限制: 65535 K
  • 问题描述
  • 农民约翰在喂奶牛的时候被另一个问题卡住了。他的所有N(1 <= N <= 100,000)个奶牛在他面前排成一行(按序号1..N的顺序),按照它们的社会等级排序。奶牛#1有最高的社会等级,奶牛#N最低。每个奶牛同时被指定了一个不唯一的附加值,这个数在0..2^21 - 1的范围内。

    帮助农民约翰找出应该从哪一头奶牛开始喂,使得从这头奶牛开始的一个连续的子序列上,奶牛的附加值的异或最大。

    如果有多个这样的子序列,选择结尾的奶牛社会等级最高的。如果还不唯一,选择最短的。

  • 输入
  • 第1行:一个单独的整数N。
    第2到N + 1行:N个0..2^21 - 1之间的整数,代表每头奶牛的被赋予的数。第j行描述了社会等级j - 1的奶牛。
  • 输出
  • 第 1 行: 3个空格隔开的整数,分别为:最大的异或值,序列的起始位置、终止位置
  • 样例输入
  • 5
    1
    0
    5
    4
    2
  • 样例输出
  • 6 4 5
  • 提示
  • 最大异或值为6,从第4个开始喂,到第5个结束。
    4 异或 2 = 6
    (100) 异或 (010) = (110)
  • 来源
  • USACO (NOCOW)

题目链接:NBUT 1525

求一段连续异或最大值,假设连续区间为[L,R]则是由pre[L-1]^pre[R]而来,然后枚举每一个pre[i]即可,每一次查找的时间复杂度为O(20)左右,然后题目就变成了求两个pre的最大异或和的情况,跟xorsum就一样了,不过要注意的是对pre[i]求出来的最优异或组合pre[j]后其实得到的是最优xorsum[i+1,j],因此左区间要加一;除此之外根据题意进行特判并且还有要考虑直接取当前值区间端点相等时候特判,调试调试就差不多了。

给几组容易错的数据:

5

5 5 6 6 7

答案为 7 5 5(题目数据不够强,之前代码答案为7 4 5 也可以AC)

5

0 1 1 0 0

答案为 1 2 2

代码:

#include <stdio.h>
#include <iostream>
#include <algorithm>
#include <cstdlib>
#include <sstream>
#include <cstring>
#include <bitset>
#include <string>
#include <deque>
#include <stack>
#include <cmath>
#include <queue>
#include <set>
#include <map>
using namespace std;
#define INF 0x3f3f3f3f
#define CLR(x,y) memset(x,y,sizeof(x))
#define LC(x) (x<<1)
#define RC(x) ((x<<1)+1)
#define MID(x,y) ((x+y)>>1)
typedef pair<int,int> pii;
typedef long long LL;
const double PI=acos(-1.0); const int N=100010; struct info
{
int val,S;
info *nxt[2];
info()
{
val=0;
S=-1;
nxt[0]=nxt[1]=NULL;
}
}; info *L;
int arr[N];
int pre[N];
void update(int n,int p)
{
bitset<21> bit(n);
info *cur=L;
int indx,i;
for (i=20; i>=0; --i)
{
indx=bit[i];
if(cur->nxt[indx])
cur=cur->nxt[indx];
else
{
info *one=new info();
cur->nxt[indx]=one;
cur=one;
}
}
cur->val=n;
if(cur->S==-1||p<cur->S)//前缀结尾位置尽量靠前
cur->S=p;
}
pii query(int n)
{
info *cur=L;
bitset<21> bit(n);
int indx,i;
for (i=20; i>=0; --i)
{
indx=bit[i];
if(cur->nxt[indx^1])
cur=cur->nxt[indx^1];
else
cur=cur->nxt[indx];
}
return pii(cur->val,cur->S);
}
int read()
{
int res=0,ch,flag=0;
if((ch=getchar())=='-')
flag=1;
else if(ch>='0'&&ch<='9')
res=ch-'0';
while((ch=getchar())>='0'&&ch<='9')
res=res*10+ch-'0';
return flag?-res:res;
}
int main(void)
{
int n,i,l,r,ans,val;
while (~scanf("%d",&n))
{
getchar();
arr[0]=0;
L=new info();
for (i=1; i<=n; ++i)
{
arr[i]=read();
pre[i]=pre[i-1]^arr[i];
update(pre[i],i);
}
int tl,tr,dx,res;
ans=0;
for (i=0; i<=n; ++i)
{
pii temp=query(pre[i]); res=max(temp.first^pre[i],arr[i]);
tl=min(temp.second+1,i);
tr=max(temp.second,i);
dx=tr-tl;
if(arr[i]==res)
{
if(i<tr)
{
tl=tr=i;
dx=0;
}
else if(i==tr&&0<dx)
{
dx=0;
tl=tr=i;
}
}
if(res>ans)
{
ans=res;
l=tl;
r=tr;
}
else if(res==ans)
{
if(r>tr)
{
r=tr;
l=tl;
}
else if(r==tr&&dx<r-l)
l=tl;
}
}
printf("%d %d %d\n",ans,l,r);
}
return 0;
}

NBUT 1525 Cow Xor(01字典树+前缀思想)的更多相关文章

  1. POJ - 3764 01字典树+前缀异或和

    异或关于前缀的特性:[u,v]=[1,u]^[1,v] 注意是路径,假设1为根,prexor[1]不保留数值 /*H E A D*/ int to[maxn<<1],nxt[maxn< ...

  2. HDU 4825 Xor Sum (模板题)【01字典树】

    <题目链接> 题目大意: 给定n个数,进行m次查找,每次查找输出n个数中与给定数异或结果最大的数. 解题分析: 01字典树模板题,01字典树在求解异或问题上十分高效.利用给定数据的二进制数 ...

  3. 51nod 1295 XOR key 可持久化01字典树

    题意 给出一个长度为\(n\)的正整数数组\(a\),再给出\(q\)个询问,每次询问给出3个数,\(L,R,X(L<=R)\).求\(a[L]\)至\(a[R]\)这\(R-L+1\)个数中, ...

  4. [BZOJ4260] Codechef REBXOR (01字典树,异或前缀和)

    Description Input 输入数据的第一行包含一个整数N,表示数组中的元素个数. 第二行包含N个整数A1,A2,-,AN. Output 输出一行包含给定表达式可能的最大值. Sample ...

  5. P4551 最长异或路径 (01字典树,异或前缀和)

    题目描述 给定一棵 n 个点的带权树,结点下标从 1 开始到 N .寻找树中找两个结点,求最长的异或路径. 异或路径指的是指两个结点之间唯一路径上的所有边权的异或. 输入输出格式 输入格式: 第一行一 ...

  6. 2014百度之星资格赛—— Xor Sum(01字典树)

    Xor Sum Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 132768/132768 K (Java/Others) Total ...

  7. Codeforces 979 D. Kuro and GCD and XOR and SUM(异或和,01字典树)

    Codeforces 979 D. Kuro and GCD and XOR and SUM 题目大意:有两种操作:①给一个数v,加入数组a中②给出三个数x,k,s:从当前数组a中找出一个数u满足 u ...

  8. Codeforces 948 数论推导 融雪前缀和二分check 01字典树带删除

    A. 全部空的放狗 B. 先O(NLOGNLOGN)处理出一个合数质因数中最大的质数是多少 因为p1 x1 x2的关系是 x2是p在x1之上的最小倍数 所以x1的范围是[x2-p+1,x2-1]要使最 ...

  9. Xor Sum---hdu4825(01字典树模板)

    题目链接:http://acm.split.hdu.edu.cn/showproblem.php?pid=4825 题意:有n个数m个查找,每个查找有一个数x, 从序列中找到一个数y,使得x异或y最大 ...

随机推荐

  1. HDU 5510 Bazinga (字符串匹配)

    题目:传送门. 题意:t 组数据,每组 n 个串,对于第 i 个串如果存在 1 到 i-1 中的某个串不是 i 的子串,那么这个第 i 个串符合题意,求 i 的最大值. 题解:KMP,AC自动机也可以 ...

  2. Docker跨主机通信之路由

    一.实验环境: 主机名 主机IP Docker0_IP Docker1 192.168.88.130 172.17.0.1 Docker2 192.168.88.131 172.18.0.1 二.实验 ...

  3. transient关键字使用笔记

    >>transient的作用及使用方法 一个对象只要实现了Serilizable接口,这个对象就可以被序列化,java的这种序列化模式为开发者提供了很多便利,我们可以不必关系具体序列化的过 ...

  4. JQZoom 图片放大插件的使用

    QZoom是一个基于最流行的jQuery的图片放大器插件.它功能强大,使用简便.支持标准模式.反转模式.无镜头.无标题的放大,并可以自定义jQZoom的窗口位置和渐隐效果,修正IE6的select b ...

  5. Visual Studio一秒变Node.js IDE

    (此文章同时发表在本人微信公众号"dotNET每日精华文章") 上个月微软发布了一个插件,可以让Visual Studio一秒变身最强大的Node.js开发工具.现在源代码移到了G ...

  6. HDU 4513 吉哥系列故事——完美队形II manacher

    吉哥系列故事——完美队形II Problem Description 吉哥又想出了一个新的完美队形游戏! 假设有n个人按顺序站在他的面前,他们的身高分别是h[1], h[2] ... h[n],吉哥希 ...

  7. [MySQL]导入导出

    [MySQL]导入导出 一 导入文本数据 1)mysql->load data infile 数据文件c:/mytable.txt 如下:(每一行为一条记录,记录的字段间用tab隔开,最后一个字 ...

  8. ember.js:使用笔记4 数组数据的分组显示

    除了之前介绍的将数组数据在一个页面中输出的方法,还可以将数组数据分组,按照点击,在不同页面中分别显示,方法为: Model: 例如:Table Router: 设置一个父对象和子对象设置: this. ...

  9. 10688 XYM-AC之路

    Description 在华农的众ACMers中,有着一位家喻户晓.人称一鸣惊人的DP神牛—XYM.由于XYM太出名了,他的仰慕者决定给XYM写一部个人传奇以传承他光辉的AC之路. 为了使故事更加真实 ...

  10. POJ2686 Traveling by Stagecoach(状压DP+SPFA)

    题目大概是给一张有向图,有n张票,每张票只能使用一次,使用一张票就能用pi匹马拉着走过图上的一条边,走过去花的时间是边权/pi,问从a点走到b点的最少时间是多少. 用dp[u][S]表示当前在u点且用 ...