题解 | #小美的区间异或和#

小美的区间异或和

https://ac.nowcoder.com/acm/problem/259733

元素a[i]对最终答案的总贡献为:a[i]在所有包含它的连续子数组中与其他元素构成的所有可能数对(a[i], a[j])的异或和。从0到n-1遍历a[i],累加每一个a[i]的贡献,就得到最终答案。遍历单个a[i]的所有包含数组需要O(n),每个数组内部计算数对异或和需要O(n^2),遍历整个a数组需要O(n),总复杂度为O(n^4),显然不行。但是通过对异或计算进行优化,可以将总复杂度降低为O(n) * O(log(max(a[i]))),满足题目要求。

对异或和的计算进行优化:注意到计算异或和时每一个二进制位是独立的,而每一位的值只有0和1两种。以第0位为例,考虑数组a中每一位元素的第0个二进制位,形成新数组a0,则"a[i]在所有包含它的连续子数组中与其他元素构成的所有可能数对(a[i], a[j])的异或和" 在第0位中的体现即为 “a0[i]在所有包含它的连续子数组中与其他元素构成的所有可能数对(a0[i], a0[j])的异或和” 。

a0[i]和a0[j]的值无非4种情况:(0,0), (0,1), (1,0), (1,1),而当a0[i]和a0[j]的值相同时二者的异或值为0,只有二者的值不同时才对结果有贡献。每一个(0,1)或(1,0)数对在第0位的贡献为1,在第b位的贡献即为1<<b。推导到这里就已经基本解决了,想办法压缩计算第b位总贡献的复杂度就行了。

对a0[i],考虑所有包含当前位置i的可能区间[l, r],区间左端点 l 的可能值为[0,i]共i+1种,区间右端点 r 的可能值为[i,n]共n-i种。如果当前元素a0[i]=0,考虑它左侧的所有a0[j]=1(j<i),累加所有(j+1)(即包含位置j的区间左端点数)与(n-i)(即位置i往右的所有可能的区间右端点数)的乘积(j+1) * (n-i)就得到了二进制位0的总贡献,(j+1)(n-i) << b 则是二进制位b的总贡献。使用两个变量sum0和sum1分别存储0和1的前缀和,实现O(n)地计算当前位的总贡献。详细步骤见代码注释。

#include <stdio.h>
#include <queue>
#include <math.h>
#include <algorithm>
#include <iostream>
#include <stack>
#define int long long
const int maxn = 1e5 + 9;
const int maxm = 1e5 + 9;
using namespace std;
const int p = 1e9 + 7;

int n;
int a[maxn];


signed main(void)
{
	ios::sync_with_stdio(0); cin.tie(0), cout.tie(0);

	cin >> n;
	for (int i = 0; i < n; i++)cin >> a[i];

	int ans = 0;

	for (int b = 0; b < 32; b++) { // 独立考虑每一位的异或
		int mask = 1 << b;
		int sum0 = 0, sum1 = 0;
		int tmp_ans = 0;
		for (int i = 0; i < n; i++) {
			if (a[i] & mask) {
				tmp_ans += sum0 * (n - i); // n-i: 能够使数组[l, r]包含a[i]的所有可能的右端点 r 的数量 (l<r)
				sum1 += (i + 1); // i+1: 能够使数组[l, r]包含a[i]的所有可能的左端点 l 的数量
				// 二者相乘表示元素a[i]在b位的总贡献,即所有包含a[i]的连续子数组中,满足b位为(0,1)或(1,0)的数对的总数量。
			}
			else {
				tmp_ans += sum1 * (n - i);
				sum0 += (i + 1);
			}
			tmp_ans %= p;
			sum0 %= p;
			sum1 %= p;
		}
		ans = (ans + (tmp_ans << b) % p) % p;
	}

	cout << ans;



	return 0;
}
全部评论

相关推荐

不愿透露姓名的神秘牛友
07-15 17:46
暑期就挂了,秋招还有机会吗
大聪明777:研发提前批,14号刚开的,官网上面的配图上有写。提前批没过的话,秋招还可以投,不过前面的笔试/面试记录会被保留,供秋招参考
26届校招投递进展
点赞 评论 收藏
分享
07-11 15:12
门头沟学院 Java
别人在上班,我就在工位上看看视频啥的,这正常吗?
程序员小白条:实习就是摸鱼,只是公司指标,把你进来了,可能那时候客户很多,但等你进来的时候,已经是淡季了,根本没多少需求,或者说根本不适合实习生去完成,因此你就每天干坐着就行,可能1,2个月都没需求
实习生的蛐蛐区
点赞 评论 收藏
分享
废物一个0offer:认真的吗二本本科找人工智能岗位
点赞 评论 收藏
分享
06-27 12:54
已编辑
门头沟学院 Java
累了,讲讲我的大学经历吧,目前在家待业。我是一个二本院校软件工程专业。最开始选专业是觉得计算机感兴趣,所以选择了他。本人学习计算机是从大二暑假结束开始的,也就是大三开始。当时每天学习,我个人认为Java以及是我生活的一部分了,就这样持续学习了一年半,来到了大四上学期末,大概是在12月中旬,我终于找的到了一家上海中厂的实习,但我发现实习生的工作很枯燥,公司分配的活也不多,大多时间也是自己在自学。就这样我秋招末才找到实习。时间来到了3月中旬,公司说我可以转正,但是转正工资只有7000,不过很稳定,不加班,双休,因为要回学校参加答辩了,同时当时也是心高气傲,认为可以找到更好的,所以放弃了转正机会,回学校准备论文。准备论文期间就也没有投递简历。然后时间来到了5月中旬,这时春招基本也结束了,然后我开始投递简历,期间只是约到了几家下场面试。工资也只有6-7k,到现在我不知道该怎么办了。已经没有当初学习的心劲了,好累呀,但是又不知道该干什么去。在家就是打游戏,boss简历投一投。每天日重一次。26秋招都说是针对26届的人,25怎么办。我好绝望。要不要参加考公、考研、央国企这些的。有没有大佬可以帮帮我。为什么感觉别人找工作都是顺其自然的事情,我感觉自己每一步都在艰难追赶。八股文背了又忘背了又忘,我每次都花很长时间去理解他,可是现在感觉八股、项目都忘完了。真的已经没有力气再去学习了。图片是我的简历,有没有大哥可以指正一下,或者说我应该走哪条路,有点不想在找工作了。
码客明:太累了就休息一下兄弟,人生不会完蛋的
如果实习可以转正,你会不...
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务