【XSY2988】取石子 博弈论
题目描述
有 \(n\) 堆石子,每堆石子的个数是 \(c_i\)。
Alice 和 Bob 轮流取石子(先后手未定),Alice 每次从一堆中取 \(a\) 个,Bob每次从一堆中取 \(b\) 个。无法操作者输。
你要选定若干堆石子(共 \(2^n\))种情况,问你所有情况中:Alice 必胜的方案数;Bob 必胜的方案数;先手必胜的方案数;后手必胜的方案数。
对 \({10}^9+7\) 取模。
\(n\leq 100000\)
题解
要把每堆石子个数对 \(a+b\) 取模。
为什么可以取模呢?
首先这是一个零和博弈。
如果 \(c_i\geq a+b\),那么一个人操作后:如果局面对另一个人更优,那么这个人就不会这么操作。否则另一个人可以取一次这堆石子,把局面变回来。
把这 \(n\) 堆按石子个数分成 \(4\) 类:
1.\(c_i<a\)
2.\(a\leq c_i<b\)
3.\(b\leq c_i<2a\)
4.\(c_i\geq 2a\)
容易发现:
第一类是没有用的。
如果有第二类,那么 \(a\) 必胜。
如果第四类石子堆的个数 \(\geq 2\),那么 \(a\) 必胜。
如果第四类石子堆的个数 \(=1\) 且 第三类石子堆的个数为奇数,则 \(a\) 必胜。
如果第四类石子堆的个数 \(=1\) 且 第三类石子堆的个数为偶数,则先手必胜。
如果第四类石子堆的个数 \(=0\) 且 第三类石子堆的个数为奇数,则先手必胜。
如果第四类石子堆的个数 \(=0\) 且 第三类石子堆的个数为偶数,则后手必胜。
每一类的方案数是 \(2\) 的幂乘上一些组合数。
时间复杂度:\(O(n)\)
代码
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cstdlib>
#include<ctime>
#include<utility>
#include<functional>
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int,int> pii;
typedef pair<ll,ll> pll;
void open(const char *s){
#ifndef ONLINE_JUDGE
char str[100];sprintf(str,"%s.in",s);freopen(str,"r",stdin);sprintf(str,"%s.out",s);freopen(str,"w",stdout);
#endif
}
int rd(){int s=0,c,b=0;while(((c=getchar())<'0'||c>'9')&&c!='-');if(c=='-'){c=getchar();b=1;}do{s=s*10+c-'0';}while((c=getchar())>='0'&&c<='9');return b?-s:s;}
void put(int x){if(!x){putchar('0');return;}static int c[20];int t=0;while(x){c[++t]=x%10;x/=10;}while(t)putchar(c[t--]+'0');}
int upmin(int &a,int b){if(b<a){a=b;return 1;}return 0;}
int upmax(int &a,int b){if(b>a){a=b;return 1;}return 0;}
const ll p=1000000007;
ll fp(ll a,ll b)
{
ll s=1;
for(;b;b>>=1,a=a*a%p)
if(b&1)
s=s*a%p;
return s;
}
int a[100010];
int A,B,n;
int cnt1,cnt2,cnt3,cnt4;
ll ans1,ans2,ans3,ans4;
ll inv[100010];
ll fac[100010];
ll ifac[100010];
ll binom(int x,int y)
{
return x>=y&&y>=0?fac[x]*ifac[y]%p*ifac[x-y]%p:0;
}
int main()
{
open("a");
scanf("%d",&n);
inv[1]=fac[0]=fac[1]=ifac[0]=ifac[1]=1;
for(int i=2;i<=n;i++)
{
inv[i]=-p/i*inv[p%i]%p;
fac[i]=fac[i-1]*i%p;
ifac[i]=ifac[i-1]*inv[i]%p;
}
int flag=0;
scanf("%d%d",&A,&B);
if(A>B)
{
swap(A,B);
flag=1;
}
for(int i=1;i<=n;i++)
{
scanf("%d",&a[i]);
a[i]%=(A+B);
if(a[i]<A)
cnt1++;
else if(a[i]<B)
cnt2++;
else if(a[i]<2*A)
cnt3++;
else
cnt4++;
}
ans1=(ans1+(fp(2,cnt2)-1)*fp(2,cnt3+cnt4))%p;
ans1=(ans1+(fp(2,cnt4)-cnt4-1)*fp(2,cnt3))%p;
ans3=(ans3+cnt4*(cnt3?fp(2,cnt3-1):1))%p;
ans1=(ans1+cnt4*(cnt3?fp(2,cnt3-1):0))%p;
ans4=(ans4+(cnt3?fp(2,cnt3-1):1))%p;
ans3=(ans3+(cnt3?fp(2,cnt3-1):0))%p;
ans1=ans1*fp(2,cnt1)%p;
ans2=ans2*fp(2,cnt1)%p;
ans3=ans3*fp(2,cnt1)%p;
ans4=ans4*fp(2,cnt1)%p;
if(flag)
swap(ans1,ans2);
ans1=(ans1+p)%p;
ans2=(ans2+p)%p;
ans3=(ans3+p)%p;
ans4=(ans4+p)%p;
printf("%lld %lld %lld %lld\n",ans1,ans2,ans3,ans4);
return 0;
}
【XSY2988】取石子 博弈论的更多相关文章
- 【ACM】取石子 - 博弈论
取石子(一) 时间限制:3000 ms | 内存限制:65535 KB 难度:2 描述 一天,TT在寝室闲着无聊,和同寝的人玩起了取石子游戏,而由于条件有限,他/她们是用旺仔小馒头当作石子.游 ...
- bzoj 3895 取石子——博弈论
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3895 看题解:https://blog.csdn.net/popoqqq/article/d ...
- bzoj 3895 取石子 —— 博弈论
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3895 看了博客:https://blog.csdn.net/popoqqq/article/ ...
- 【BZOJ1413】[ZJOI2009]取石子游戏(博弈论,动态规划)
[BZOJ1413][ZJOI2009]取石子游戏(博弈论,动态规划) 题面 BZOJ 洛谷 题解 神仙题.jpg.\(ZJOI\)是真的神仙. 发现\(SG\)函数等东西完全找不到规律,无奈只能翻题 ...
- POJ.1067 取石子游戏 (博弈论 威佐夫博弈)
POJ.1067 取石子游戏 (博弈论 威佐夫博弈) 题意分析 简单的威佐夫博弈 博弈论快速入门 代码总览 #include <cstdio> #include <cmath> ...
- HDU.2516 取石子游戏 (博弈论 斐波那契博弈)
HDU.2516 取石子游戏 (博弈论 斐波那契博弈) 题意分析 简单的斐波那契博弈 博弈论快速入门 代码总览 #include <bits/stdc++.h> #define nmax ...
- bzoj3895: 取石子(博弈论,记忆化搜索)
3895: 取石子 Time Limit: 1 Sec Memory Limit: 512 MBSubmit: 361 Solved: 177[Submit][Status][Discuss] D ...
- 取石子游戏 HDU 1527 博弈论 威佐夫博弈
取石子游戏 HDU 1527 博弈论 威佐夫博弈 题意 有两堆石子,数量任意,可以不同.游戏开始由两个人轮流取石子.游戏规定,每次有两种不同的取法,一是可以在任意的一堆中取走任意多的石子:二是可以在两 ...
- HDU 2516 取石子游戏 (博弈论)
取石子游戏 Problem Description 1堆石子有n个,两人轮流取.先取者第1次能够取随意多个,但不能所有取完.以后每次取的石子数不能超过上次取子数的2倍.取完者胜.先取者负输出" ...
随机推荐
- 微信小程序(四) 模板的使用
模板的使用:单独建立一个页面,在另一个页面通过name属性名调用使用(注意要导入模板路径) template.wxml页面
- asp.net core参数保护之自定义要保护的参数类型
asp.net core参数保护之自定义要保护的参数类型 Intro 为了实现 asp.net core 下的参数保护,扩展了asp.net core 中 DataProtection,可以自动化的保 ...
- 借助表达式树感受不一样的CRUD
借助表达式树感受不一样的CRUD Intro 最近有个想法,想不写 sql 语句,做一个类似于 ORM 的东西,自己解析表达式树,生成要执行的 sql 语句,最后再执行 sql 语句,返回相应结果. ...
- Oracle database link中查询会开启事务吗?
关于oracle database link,使用database link相关的查询语句是否会开启事务呢?我们知道,在数据库中一个简单的SELECT查询语句不会产生事务(select for upd ...
- logback日志配置
第一步:加入jar包.要加入slf4j和logback的jar包,slf4j需要的jar包为slf4j-api,logback需要2个jar包(logback-classic.logback-core ...
- Zabbix常见触发器表达式
Zabbix trigger是zabbix 进行告警通知的设定条件 ,当监控获取的值触发了设定的条件时,会按照触发器的设定,执行相应的action 操作 .在zabbix中为了比较方便的设定各种条件, ...
- 文件操作命令(move)
move命令: // 描述: 将一个或多个文件从一个目录移动到另一个目录. // 语法: move [{/y | /-y}] [<Source>] [<Target>] // ...
- 大数据平台Lambda架构详解
Lambda架构由Storm的作者Nathan Marz提出.旨在设计出一个能满足.实时大数据系统关键特性的架构,具有高容错.低延时和可扩展等特. Lambda架构整合离线计算和实时计算,融合不可变( ...
- @FeignClient
@FeignClient("APP-PROVIDER")public interface MyFeignClient { @RequestMapping(value = " ...
- 替换空格[by Python]
题目: 请实现一个函数,将一个字符串中的空格替换成“%20”.例如,当字符串为We Are Happy.则经过替换之后的字符串为We%20Are%20Happy. 1.使用python自带的repla ...