【CF662A】Gambling Nim 线性基

【CF662A】Gambling Nim

题意:n长卡牌,第i张卡牌正面的数字是$a_i$,反面的数字是$b_i$,每张卡牌等概率为正面朝上或反面朝上。现在Alice和Bob要用每张卡牌朝上的数字玩NIM游戏,问先手获胜的概率。

$n\le 5000,a_i,b_i\le 10^{18}$

题解:傻逼题都不会了,先令所有的都是正面朝上,再令$S=a_1\ \text{xor}\ a_2...a_n,c_i=a_i\ \text{xor}\ b_i$,则问题变成了选出一些$c_i$使得异或和为$S$的概率。显然搞基一发,然后将S放到线性基里消一下。如果能消没,则概率为$1-{1\over 2}^{siz}$,siz是线性基大小。否则概率是1。

#include <cstdio>
#include <cstring>
#include <iostream>

using namespace std;
typedef long long ll;
const int maxn=500010;
int n,m;
ll S,v[maxn];
inline ll rd()
{
	ll ret=0,f=1;	char gc=getchar();
	while(gc<'0'||gc>'9')	{if(gc=='-')	f=-f;	gc=getchar();}
	while(gc>='0'&&gc<='9')	ret=ret*10+(gc^'0'),gc=getchar();
	return ret*f;
}
int main()
{
	n=rd();
	int i,j;
	ll a,b;
	for(i=0;i<n;i++)	a=rd(),b=rd(),S^=a,v[i]=a^b;
	for(i=60;i>=0;i--)
	{
		for(j=m;j<n;j++)	if((v[j]>>i)&1)	break;
		if(!((v[j]>>i)&1))	continue;
		if(m!=j)	swap(v[m],v[j]);
		for(j=0;j<n;j++)	if(j!=m&&((v[j]>>i)&1))	v[j]^=v[m];
		m++;
	}
	for(i=0;i<m;i++)	if((S^v[i])<S)	S^=v[i];
	if(S)	puts("1/1");
	else	printf("%lld/%lld",(1ll<<m)-1,1ll<<m);
	return 0;
}//4 1 2 2 4 4 8 8 1

转载于:https://www.cnblogs.com/CQzhangyu/p/8594566.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值