BZOJ 2844: albus就是要第一个出场 [高斯消元XOR 线性基]

时间:2023-03-09 01:40:53
BZOJ 2844: albus就是要第一个出场 [高斯消元XOR 线性基]

2844: albus就是要第一个出场


题意:给定一个n个数的集合S和一个数x,求x在S的$2^n$个子集从小到大的异或和序列中最早出现的位置


一开始看错题了...人家要求的是x第一次出现位置不是第x个是谁

求出线性基后我们知道一共有$2^r$个不同的数,再知道每个数出现了几次就好啦

每个数出现了$2^{n-r}$次....因为有$n-r$个线性相关(高斯消元后全0了)的方程异或不影响....

然后就简单了,从高到低枚举二进制位,异或这一位后小于k就加上

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
#include <bitset>
using namespace std;
typedef long long ll;
const int N=1e5+,INF=1e9,P=;
inline int read(){
char c=getchar();int x=,f=;
while(c<''||c>''){if(c=='-')f=-;c=getchar();}
while(c>=''&&c<=''){x=x*+c-'';c=getchar();}
return x*f;
}
int n,x;
int a[N],bin[];
void ini(){
bin[]=;for(int i=;i<=;i++) bin[i]=bin[i-]<<;
}
int now;
void Gauss(){
now=;
for(int i=;i>=;i--){
int j=now;
while(j<=n&&!(a[j]&bin[i])) j++;
if(j==n+) continue;
if(j!=now) swap(a[j],a[now]);
for(int k=;k<=n;k++)
if(k!=now&&(a[k]&bin[i])) a[k]^=a[now];
now++;
}
now--;
}
int main(){
freopen("in","r",stdin);
ini();
n=read();
for(int i=;i<=n;i++) a[i]=read();
Gauss();
x=read();
int val=,sum=;
for(int i=;i<=now;i++) if((val^a[i])<=x){
val^=a[i];
sum=(sum+(<<(now-i))%P)%P;
}
for(int i=;i<=n-now;i++) sum=(sum<<)%P;
printf("%d",(sum+)%P);
}