树状数组【CF703D】Mishka and Interesting sum

Description

给你n( 1<=n<=1000000)个数,以及m(1<=m<=1000000)个询问,每个询问包括l和r,问你在这n个数中,区间l~r,出现偶数个数的数的异或和

Input

第一行一个整数 n,表示数列的长度

接下来一行 n 个非负整数,表示 a 数组中的每个元素

接下来一行一个整数 m,表示查询的数量

接下来 m 行,每行两个整数 l, r 表示这次查询区间的左右端点

Output

对于每组查询,输出一行一个整数,表示这组查询的答案

刚看到题的一瞬间,输出出现偶数次的数的异或和.

"不是0嘛?这不sb题?"

突然发现看错题.

原来是求出现偶数次的单个数的异或和

通过一些推导可以发现,答案是求区间内不同数的异或和与区间异或和的异或和

为什么?

我们假设当前查询的区间的数为:(1,3,4,2,5,4,1)

此时根据异或的性质(x) ^(x=0),我们再异或上区间内不同的数。

则这段异或起来就是:(1)^(4)

稍作解释一下

我们区间中出现偶数次的数的异或和就是(0),此时再异或上区间内不同的数。

此时这些出现偶数次的数的异或再次出现。

而那些出现单次的数就消失了.

那么现在问题就变为维护区间内不同的数的异或和

这题数据范围的话,需要离散化

因为没有修改操作,所以考虑离线.我们对右端点进行排序(从小到大)

然后考虑用一种数据结构维护:线段树 or 树状数组。

这里用了树状数组

树状数组维护异或

记录一下这个数上一个出现的位置.

然后遇到某个位置再出现,我们再在树状数组删去这个数几个(即再异或一次.)

代码

#include<cstdio>
#include<iostream>
#include<algorithm>
#define R register

using namespace std;

const int gz=3000008;

inline void in(R int &x)
{
	R int f=1;x=0;R char s=getchar();
	while(!isdigit(s)){if(s=='-')f=-1;s=getchar();}
	while(isdigit(s)){x=x*10+s-'0';s=getchar();}
	x*=f;
}

int n,a[gz],sum[gz],pre[gz],b[gz],head[gz];

int new_n=1,q,ans[gz];

struct cod
{
	int l,r,idx;
	bool operator <(const cod&a)const
	{
		return r<a.r;
	}
}que[gz];

#define lowbit(x) x&-x

int tr[gz<<1];

inline void add(R int o,R int del)
{
	for(;o<=n;o+=lowbit(o))
		tr[o]^=del;
}

inline int query(R int o)
{
	R int res=0;
	for(;o;o-=lowbit(o))
		res^=tr[o];
	return res;
}

int main()
{
	in(n);
	for(R int i=1;i<=n;i++)in(a[i]),b[i]=a[i];
	sort(b+1,b+n+1);
	for(R int i=2;i<=n;i++)
		if(b[new_n]!=b[i])b[++new_n]=b[i];
	for(R int i=1;i<=n;i++)
		a[i]=lower_bound(b+1,b+new_n+1,a[i])-b;
	for(R int i=1;i<=n;i++)
	{
		sum[i]=sum[i-1]^b[a[i]];
		pre[i]=head[a[i]];
		head[a[i]]=i;
	}
	in(q);
	for(R int i=1;i<=q;i++)
		in(que[i].l),in(que[i].r),que[i].idx=i;
	sort(que+1,que+q+1);
	R int now=1;
	for(R int i=1;i<=q;i++)
	{
		R int r=que[i].r,l=que[i].l;
		while(now<=r)
		{
			if(pre[now])
				add(pre[now],b[a[now]]);
			add(now,b[a[now]]);
			now++;
		}
		ans[que[i].idx]=(query(r)^query(l-1))^(sum[r]^sum[l-1]);
	}
	for(R int i=1;i<=q;i++)
		printf("%d
",ans[i]);
	
	
	return 0;
}

原文地址:https://www.cnblogs.com/-guz/p/9897968.html