题意

已知一个长度为 n 的整数数列 a[1],a[2],…,a[n] ,给定查询参数 l、r ,问在 [l,r] 区间内,有多少连续子

序列满足异或和等于 k 。

也就是说,对于所有的 x,y (l≤x≤y≤r),能够满足a[x]a[x+1]…^a[y]=k的x,y有多少组。

分析

这样的题目首先按照异或运算前缀和,就变成多次查询区间内有多少对数满足异或和为k.

考虑简单的情况.异或和为0的时候,就变成查询区间内有多少对数相同.这是很显然的莫队题目.

那么异或和为k的时候我们也可以考虑莫队.比较简明的思路是用trie树维护区间内所有的数字,加入/删除数字的时候更新答案.

另一种方式是,把所有的前缀和异或上k,组成另外一个数列,那么对于询问的区间在原先的前缀和数列和异或k的前缀和数列上都有一个数集,然后找两个数集的相同的数对(就是找这样的数对:数值相同,但是一个数在这个数集,另一个数在那个数集).

#include<cstdio>
#include<algorithm>
using namespace std;
const int maxn=100005;
int a[maxn],s1[maxn],s2[maxn];
int cnt1[500000],cnt2[500000],ans;
int SZ=250;
struct query{
int l,r,num,ans;
void read(){
scanf("%d%d",&l,&r);l--;
}
}Q[maxn];
bool cmp1(const query &A,const query &B){
if(A.l/SZ!=B.l/SZ)return A.l<B.l;
return A.r<B.r;
}
void init(int l,int r){
//printf("%d %d\n",l,r); for(int i=l;i<=r;++i){
//printf("%d ",s1[i]);
cnt1[s1[i]]++;
}//printf("\n");
for(int i=l;i<=r;++i){
cnt2[s2[i]]++;//printf("%d ",s2[i]);
ans+=cnt1[s2[i]];
}//printf("\n");
//printf("%d\n",ans);
}
void move(int l1,int r1,int l2,int r2){
for(int i=l1;i<l2;++i){
ans-=cnt2[s1[i]];ans-=cnt1[s2[i]];
cnt1[s1[i]]--;cnt2[s2[i]]--;
if(s1[i]==s2[i])ans+=1;
}
for(int i=l1-1;i>=l2;--i){
ans+=cnt2[s1[i]];ans+=cnt1[s2[i]];
cnt1[s1[i]]++;cnt2[s2[i]]++;
if(s1[i]==s2[i])ans-=1;
}
for(int i=r1+1;i<=r2;++i){
ans+=cnt2[s1[i]];ans+=cnt1[s2[i]];
cnt1[s1[i]]++;cnt2[s2[i]]++;
if(s1[i]==s2[i])ans-=1;
}
for(int i=r1;i>r2;--i){
ans-=cnt2[s1[i]];ans-=cnt1[s2[i]];
cnt1[s1[i]]--;cnt2[s2[i]]--;
if(s1[i]==s2[i])ans+=1;
}
}
int res[maxn];
int main(){
int n,m,k;scanf("%d%d%d",&n,&m,&k);
for(int i=1;i<=n;++i){
scanf("%d",a+i);
}
for(int i=1;i<=n;++i)s1[i]=s1[i-1]^a[i];
for(int i=0;i<=n;++i)s2[i]=s1[i]^k;
for(int i=1;i<=m;++i)Q[i].read();
for(int i=1;i<=m;++i)Q[i].num=i;
sort(Q+1,Q+m+1,cmp1);
init(Q[1].l,Q[1].r);Q[1].ans=ans;
for(int i=2;i<=m;++i){
move(Q[i-1].l,Q[i-1].r,Q[i].l,Q[i].r);
Q[i].ans=ans;
}
if(k==0){
for(int i=1;i<=m;++i)Q[i].ans-=(Q[i].r-Q[i].l+1);
}
for(int i=1;i<=m;++i)Q[i].ans/=2;
for(int i=1;i<=m;++i)res[Q[i].num]=Q[i].ans;
for(int i=1;i<=m;++i)printf("%d\n",res[i]);
return 0;
}

bzoj5301[CQOI2018]异或序列的更多相关文章

  1. BZOJ5301: [Cqoi2018]异或序列(莫队)

    5301: [Cqoi2018]异或序列 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 400  Solved: 291[Submit][Status ...

  2. [bzoj5301][Cqoi2018]异或序列_莫队

    异或序列 bzoj-5301 Cqoi-2018 题目大意:题目链接. 注释:略. 想法: 由于a^a=0这个性质,我们将所有的数变成异或前缀和. 所求就变成了求所有的$l_i\le x<y\l ...

  3. BZOJ5301:[CQOI2018]异或序列(莫队)

    Description 已知一个长度为 n 的整数数列 a[1],a[2],…,a[n] ,给定查询参数 l.r ,问在 [l,r] 区间内,有多少连续子 序列满足异或和等于 k . 也就是说,对于所 ...

  4. 2018.08.12 bzoj5301: [Cqoi2018]异或序列(前缀和+莫队)

    传送门 简单的异或前缀和处理+莫队统计答案. 惊奇的发现无论开不开long long都能跑过... 代码: #include<bits/stdc++.h> #define N 100005 ...

  5. BZOJ5301 [Cqoi2018]异或序列 【莫队】

    题目链接 BZOJ5301 题解 莫队水题 BZOJ400AC纪念 #include<algorithm> #include<iostream> #include<cst ...

  6. bzoj 5301: [Cqoi2018]异或序列 (莫队算法)

    链接:https://www.lydsy.com/JudgeOnline/problem.php?id=5301 题面; 5301: [Cqoi2018]异或序列 Time Limit: 10 Sec ...

  7. 「luogu4462」[CQOI2018] 异或序列

    「luogu4462」[CQOI2018]异或序列 一句话题意 输入 \(n\) 个数,给定\(k\),共 \(m\) 组询问,输出第 \(i\) 组询问 \(l_i\) \(r_i\) 中有多少个连 ...

  8. bzoj 5301 [Cqoi2018]异或序列 莫队

    5301: [Cqoi2018]异或序列 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 204  Solved: 155[Submit][Status ...

  9. 并不对劲的复健训练-bzoj5301:loj2534:p4462 [CQOI2018]异或序列

    题目大意 给出一个序列\(a_1,...,a_n\)(\(a,n\leq 10^5\)),一个数\(k\)(\(k\leq 10^5\)),\(m\)(\(m\leq10^5\))次询问,每次询问给\ ...

随机推荐

  1. 重写Override和重加载Overload

    1.方法的重写规则 参数列表必须完全与被重写方法的相同: 返回类型必须完全与被重写方法的返回类型相同: 访问权限不能比父类中被重写的方法的访问权限更低.例如:如果父类的一个方法被声明为public,那 ...

  2. js中数组的使用

    使用js时候,很多情况下要使用数组.就写写数组中一些常用的方法. js中定义一个数组,一般有以下2种方法. 1. var arr = new Array(3); // 声明数组有3个元素 arr[0] ...

  3. Centos 发送smtp邮件

    说明:          1.本文是用网易smtp服务,QQ的没试过        2.在Centos7上测试 实现:        1.关闭本机的sendmail服务或者postfix服务     ...

  4. Ubuntu+Qt+OpenCV+FFMPEG环境搭建

    基于ubuntu16.04下opencv3.2安装配置 Ubuntu16.04下安装FFmpeg(超简单版) Qt编译后提示: /usr/bin/ld: 找不到 -lGL 安装libGL: sudo ...

  5. eclipse 最最最常用快捷键

    使用eclipse这么久,发现其跟PS一样,使用一些快捷键会有效率很多. 至此总结出以下每次打开eclipse基本都会用上的快捷键. 不熟悉这些快捷键,在实际编程中有意识使用的话对以后编码很有帮助. ...

  6. linux下的tar命令详解

    通过SSH访问服务器,难免会要用到压缩,解压缩,打包,解包等,这时候tar命令就是是必不可少的一个功能强大的工具.linux中最流行的tar是麻雀虽小,五脏俱全,功能强大. tar命令可以为linux ...

  7. 搭建django虚拟环境完整步骤

    一.建立虚拟环境 pip install virtualenv 要使用Django,首先要建立一个虚拟工作环境.我们先为项目建立一个文件夹learn,在文件夹中打开命令行(shift+右击),来建立另 ...

  8. 基于.NET Standard的分布式自增ID算法--Snowflake代码实现

    概述 上篇文章介绍了3种常见的Id生成算法,本篇主要介绍如何使用C#实现Snowflake. 基础字段 /// <summary> /// 工作节点Id(长度为5位) /// </s ...

  9. grunt-inline:一个资源内嵌插件

    一.插件简介 将引用的外部资源,如js.css.img等,内嵌到引用它们的文件里去. 二.使用场景 在项目中,出于某些原因,有的时候我们需要将一些资源,比如js脚本内嵌到页面中去.比如我们的html页 ...

  10. 阿里云 ECS 监控报警设置

    1.阿里云监控项说明 https://helpcdn.aliyun.com/document_detail/43505.html 2.监控设置 3.报警规则 4.设置阈值 5.确定即可. 6.效果图