第十四届蓝桥杯省赛 C/C++ A 组 H 题——异或和之和(AC)

2023-07-13 22:02:55

1. 异或和之和

1.题目描述

给定一个数组 A i A_i Ai,分别求其每个子段的异或和,并求出它们的和。或者说,对于每组满足 1 ≤ L ≤ R ≤ n 1 \leq L \leq R \leq n 1LRn L , R L, R L,R,求出数组中第 L L L 至第 R R R 个元素的异或和。然后输出每组 L , R L, R L,R 得到的结果加起来的值。

2.输入格式

输入的第一行包含一个整数 n n n

第二行包含 n n n 个整数 A i A_i Ai,相邻整数之间使用一个空格分隔。

3.输出格式

输出一行包含一个整数表示答案。

4.样例输入

5
1 2 3 4 5

5.样例输出

39

6. 数据范围

对于 30 % 30\% 30% 的评测用例, n ≤ 300 n \leq 300 n300

对于 60 % 60\% 60% 的评测用例, n ≤ 5000 n \leq 5000 n5000

对于所有评测用例, 1 ≤ n ≤ 1 0 5 1 \leq n \leq 10^5 1n105 0 ≤ A i ≤ 2 20 0 \leq A_i \leq 2^{20} 0Ai220

7. 原题链接

异或和之和

2. 解题思路

首先,根据题意进行暴力枚举每个子区间 [ l , r ] [l,r] [l,r] ( 1 ≤ l ≤ r ≤ n ) (1\leq l \leq r \leq n) (1lrn) 的异或和,复杂度将是 O ( n 2 ) O(n^2) O(n2),无法通过本题,但可以拿到一定分数。

因为涉及异或运算且观察到 a i a_i ai 的取值范围为 [ 0 , 2 20 ] [0,2^{20}] [0,220],我们不难想到从"拆位"的角度去思考问题。假设对于二进制位的第 i ∈ [ 0 , 20 ] i \in[0,20] i[0,20] 位,数组中一共有 x x x 个子区间在该位的异或和为 1 1 1,那么该位对答案的贡献为 2 i × x 2^{i} \times x 2i×x。这样我们就将整个大问题,拆成了 20 20 20 个子问题,原数组相当于被我们拆分为 20 20 20 01 01 01 数组。

对于每个子问题,也就是对于每个 01 01 01 数组,我们需要求出有多少子数组的异或和为 1 1 1,也就是求出前面所说的 x x x。这个问题我们可以通过前缀异或来解决。设这里的 01 01 01 数组为 a a a 数组,我们再设一个 S S S 数组, S i S_i Si 表示 a a a 数组中前 i i i 个元素的异或和为多少。根据异或运算的性质:

a i ⊕ a i + 1 ⊕ ⋯ ⊕ a j − 1 ⊕ a j = ( a 1 ⊕ a 2 ⊕ ⋯ a j ) ⊕ ( a 1 ⊕ a 2 ⊕ ⋯ ⊕ a i − 1 ) a_i \oplus a_{i+1} \oplus \cdots \oplus a_{j-1} \oplus a_j=(a_1 \oplus a_2 \oplus \cdots a_j) \oplus (a_1 \oplus a_2 \oplus \cdots \oplus a_{i-1}) aiai+1aj1aj=(a1a2aj)(a1a2ai1)

将上述式子用 S S S 进行代替有:

a i ⊕ a i + 1 ⊕ ⋯ ⊕ a j − 1 ⊕ a j = S i − 1 ⊕ S j a_i \oplus a_{i+1} \oplus \cdots \oplus a_{j-1} \oplus a_j=S_{i-1} \oplus S_j aiai+1aj1aj=Si1Sj

如果想使得等式左边等于 1 1 1,即需要满足 S i − 1 ≠ S j S_{i-1} \ne S_j Si1=Sj

根据上述分析,我们可以枚举每个 01 01 01 数组的前缀异或数组 S S S,当我们枚举到 S j S_j Sj 时,我们只需要考虑在它之前有多少个 S i S_i Si 和它不同,不同的个数就等于以 a j a_j aj 结尾且异或和为 1 1 1 的子数组个数,这个统计个数的功能我们可以使用哈希表实现。

这样我们可以在接近 O ( n ) O(n) O(n) 的复杂度统计出每个 01 01 01 数组子区间异或为 1 1 1 的个数,我们设第 i i i 个二进制位的 01 01 01 数组子区间异或为 1 1 1 的个数为 b i b_i bi,最终答案为:

∑ i = 0 20 2 i × b i \sum_{i=0}^{20} 2^{i} \times b_i i=0202i×bi

时间复杂度: O ( 20 × n ) O(20 \times n) O(20×n)

3. AC_Code

  • C++
#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
const int N = 100010;

int n;
int a[N][25];
int main()
{
	ios_base :: sync_with_stdio(false);
	cin.tie(0); cout.tie(0);
	cin >> n;
	for (int i = 1; i <= n; ++i) {
		int x;
		cin >> x;
		for (int j = 0; j <= 20; ++j) {
			a[i][j] = (x >> j) & 1;
			a[i][j] ^= a[i - 1][j];
		}
	}
	LL ans = 0;
	for (int j = 0; j <= 20; ++j) {
		map<int, int> m;
		m[0]++;
		for (int i = 1; i <= n; ++i) {
			int x = m[a[i][j] ^ 1];
			ans += 1LL * (1 << j) * x;
			m[a[i][j]]++;
		}
	}
	cout << ans << '\n';
	return 0;
}
  • Java
import java.util.*;

public class Main {
    static final int N = 100010;

    public static void main(String[] args) {
        Scanner scan = new Scanner(System.in);
        int n = scan.nextInt();
        int[][] a = new int[N][25];
        for (int i = 1; i <= n; ++i) {
            int x = scan.nextInt();
            for (int j = 0; j <= 20; ++j) {
                a[i][j] = (x >> j) & 1;
                a[i][j] ^= a[i - 1][j];
            }
        }
        long ans = 0;
        for (int j = 0; j <= 20; ++j) {
            Map<Integer, Integer> m = new HashMap<>();
            m.put(0, 1);
            for (int i = 1; i <= n; ++i) {
                int x = m.getOrDefault(a[i][j] ^ 1, 0);
                ans += (1L << j) * x;
                m.put(a[i][j], m.getOrDefault(a[i][j], 0) + 1);
            }
        }
        System.out.println(ans);
    }
}
  • Python
n = int(input())
N = 100010
a = [[0] * 25 for _ in range(N)]
b = list(map(int, input().split()))
for i in range(1, n + 1):
    x = b[i-1]
    for j in range(21, -1, -1):
        a[i][j] = (x >> j) & 1
        a[i][j] ^= a[i - 1][j]
ans = 0
for j in range(21, -1, -1):
    m = {0: 1}
    for i in range(1, n + 1):
        x = m.get(a[i][j] ^ 1, 0)
        ans += (1 << j) * x
        m[a[i][j]] = m.get(a[i][j], 0) + 1
print(ans)

更多推荐

2023年华为杯研究生数学建模竞赛辅导

2023年华为杯研究生数学建模竞赛辅导各研究生培养单位:中国研究生数学建模竞赛作为教育部学位管理与研究生教育司指导,中国学位与研究生教育学会、中国科协青少年科技中心主办的“中国研究生创新实践系列大赛”主题赛事之一,是一项面向在校研究生进行数学建模应用研究与实践的学术竞赛活动,是广大在校研究生提高建立数学模型和运用互联网

Linux多线程【线程控制】

✨个人主页:北海🎉所属专栏:Linux学习之旅🎃操作环境:CentOS7.6阿里云远程服务器文章目录🌇前言🏙️正文1、线程知识补充1.2、线程私有资源1.3、线程共享资源1.4、原生线程库2、线程控制接口2.1、线程创建2.1.1、一批线程2.2、线程等待2.3、线程终止2.4、线程实战2.5、其他接口2.5.

Flutter动态化开发之Fair实战

一、背景目前移动端应用的版本更新,最常见的方式是定期发版,无论是安卓还是iOS,都需要提交新的安装包到应用市场进行审核。审核通过后,用户在应用市场进行App的下载更新。而动态化,就是不依赖更新程序安装包,就能动态实时更新页面的技术。相比动态化技术,定期发版更新应用的方式存在一些问题,比如:审核周期长,且可能审核不通过。

【完全二叉树魔法:顺序结构实现堆的奇象】

本章重点二叉树的顺序结构堆的概念及结构堆的实现堆的调整算法堆的创建堆排序TOP-K问题1.二叉树的顺序结构普通的二叉树是不适合用数组来存储的,因为可能会存在大量的空间浪费。而完全二叉树更适合使用顺序结构存储。现实中我们通常把堆(一种二叉树)使用顺序结构的数组来存储,需要注意的是这里的堆和操作系统虚拟进程地址空间中的堆是

PostgreSQL缓存管理

缓冲区管理器、存储和后端进程之间的关系缓存管理结构PostgreSQL缓冲区管理器由buffertable、bufferdescriptors和bufferpool组成。bufferpool层存储表和索引等数据文件页,以及空闲空间映射和可见性映射。bufferpool是一个数组,每个槽存储数据文件的一个页面。缓冲池数组

No suitable driver found for jdbc:mysql://localhost:3306/BookManagement

目录问题背景解决办法问题背景今天遇见一个这个报错,解决后将解决方案分享一下:报错内容如下:“"C:\ProgramFiles\Java\jdk1.8.0_221\bin\java.exe""-javaagent:D:\IDEA2020_1\IntelliJIDEA2020.1\lib\idea_rt.jar=51910

React 全栈体系(六)

第三章:React应用(基于React脚手架)二、组件的组合使用-TodoList3.添加todo3.1App/*src/App.jsx*///创建“外壳”组件AppimportReact,{Component}from'react'importHeaderfrom'./components/Header'import

服务器数据恢复-ESX SERVER常见故障的数据恢复的可能性分析

ESXSERVER常见故障表现:1、因光纤存储设备连接至非ESX环境,共享未互斥,对存储进行的改写操作(如:重装系统,WINDOWS初始化,格式化等)导致存储结构损坏。2、卷升级、变更时分区表或VMFS卷结构异常。3、VMFS存储中VMDK被删除。4、VMFS被格式化。ESXSERVER故障解决方案:1、检测是否存在硬

[X3m]Ubuntu 根文件系统制作

使用ubuntu20.04sudoapt-getinstallwgetca-certificatesdevice-tree-compilerpvbclzopzipbinfmt-support\build-essentialccachedebootstrapntpdategawkgcc-arm-linux-gnueabi

【Django】 rest_framework接口开发流程及接口组成

rest_framework接口开发流程及接口组成使用restframework框架开发接口,方式应该有6、7种,每个人的习惯不同,用的方法也不一样,再次不再一一详述。我比较常用:ModelSerializer+GenericAPIView原因是用视图函数+装饰器、视图类+继承APIView、或者混入Mixin这三种封

React 窗口防抖

假如有这种需求:浏览器的窗口不断缩小变大,此时页面的布局不会自动刷新,需要手动刷新页面才会自适应大小。此时我们立马就会想到使用windows的onresize方法window.onresize=()=>{//重新渲染画面root.render(<App/>)}但是新的问题就会出现这个onresize方法会被调用多次,多

热文推荐