CF - D. Ehab and the Expected XOR Problem(前缀异或)

https://codeforces.com/contest/1174/problem/D

题意
给定两个数 n 和 x,构造一个满足下列条件的数组 a[]:

  • 1 ≤ a i < 2 n 1 \le a_i<2^n 1ai<2n
  • 不存在某个非空子段的异或和为 0 或者 x;
  • 数组长度尽可能长。

1 ≤ n ≤ 18 ,   1 ≤ x < 2 18 1 \le n \le 18,\ 1 \le x<2^{18} 1n18, 1x<218

思路
如果已经构造出前 n-1 个位置,那么对于第 n 个位置来说:

  • 前 n 个位置的异或和 s[n] 不能在前面某个位置作为前缀异或和出现过;
    如果在第 t 个位置出现过,那么区间 [t+1, n-1] 这段区间的异或和就为 0,不满足。
  • 前 n 个位置的异或和 s[n] ^ x 不能再前面某个位置作为前缀异或和出现过;
    如果出现过,区间 [t+1, n-1] 的区间异或和就为 x,不满足。

假设当前位置的前缀异或为 y,那么一定要保证 yy^x 没在前面出现过。所以 yy^x 只能有一个作为前缀异或出现在答案数组中,两两配对,和其他值互不影响。
所以这些作为前缀异或的值的顺序没有要求,随意确定一种顺序即可。

如果一种前缀异或的性质确定了,那么原数组也就确定了:s[i] ^ s[i-1] 便是 a[i]

所以对于每个位置,可以从小到大枚举前缀异或 s[i],如果当前值在之前没有作为前缀异或出现过,并且其异或上 x 也没有出现过,那么这个值就可以作为前缀异或。

如果每个位置都枚举一遍 2 18 2^{18} 218 个数的话,复杂度很高。
发现如果当前枚举的值不满足要求的话,这个值在后面的位置一定用不到,因为一直要保证这个值在前面没有出现。所以可以用一个指针指向枚举的数,单调往后走。

Code

#include<bits/stdc++.h>
using namespace std;

#define Ios ios::sync_with_stdio(false),cin.tie(0)
#define int long long

const int N = 2000010, mod = 1e9+7;
int T, n, m;
int a[N], b[N];

signed main(){
	Ios;
	int x;
	cin >> n >> x;
	
	int now = 0, idx = 0;
	
	mp[0] = 1;
	while(now + 1 < 1<<n)
	{
		now ++;
		if(mp[now] || mp[now ^ x]) continue;
		a[++idx] = now;
		b[idx] = now ^ a[idx - 1];
		mp[now] = 1;
	}
	
	cout << idx << endl;
	for(int i=1;i<=idx;i++) cout << b[i] << " ";
	
	return 0;
}

经验

  • 对于连续区间的异或值有要求,要想到前缀异或。
  • 毕竟异或的负负得正性质 x ^ y ^ y = x
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值