【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倍.取完者胜.先取者负输出" ...
随机推荐
- springboot项目打包运行
在springboot项目打包成jar包时,在cmd中使用java -jar **.jar时,浏览器无法访问. 解决方法:把项目改成war包项目,在pom文件中更改,并打成war包. 使用maven命 ...
- Android远程桌面助手扩展之微信跳一跳辅助
微信跳一跳的外挂辅助已是五花八门,万能的TB上也有了各种明码标价的代练.微信小程序游戏的火爆甚至带火了手游外挂产业.另一方面,跳一跳游戏也在不断更新,防止使用外挂刷高分.Android远程桌面助手支持 ...
- java新知识系列 五
类方法和对象方法的使用限制 abstract修饰符的注意 静态变量只能在类主体中定义,不能在方法中定义 线程的各种方法差别 关于抽象类 什么是中间件 Servlet生命周期的三个主要方法 可以修饰类的 ...
- FT 软件项目管理
FT 软件项目: 以Feature Team形式组织起来的软件研发项目. 项目是临时组织不是长期组织. 人员临时组织起来, 无组织汇报关系.大家需要充分理解和认同项目的目标,通过项目获得技术.经验. ...
- Hasse神舟笔记本卡logo解决,刷BIOS方法,教你修复神船
我的电脑是神舟战神K660E i7 d7的,前两天装Windows10,Ubuntu,MAC OS Mojave,PE 一堆操作,使用bootice重建uefi引导,结果在前几天,我删了一个重复的ue ...
- MySQL下perror工具查看System Error Code信息
在MySQL数据库的维护过程中,我们有时候会在MySQL的错误日志文件中看到一些关于Operating system error的错误信息,例如在MySQL的错误日志里面,有时候会看到关于 Inn ...
- 云服务器挂载/dev/vdb1磁盘
1.首先检查是否有一块硬盘还未被挂载 2.如图所示,vdb还未被使用,开始分配它吧 [root@localhost ~]# fdisk /dev/vdb Command (m for help): n ...
- 【原】Java学习笔记019 - 面向对象
package cn.temptation; public class Sample01 { public static void main(String[] args) { // 仔细想一想,Ani ...
- java反射(java.lang.reflect)---java.lang.reflect.Modifier中状态码
1. 详情请看jvm(虚拟机)规范 java.lang.reflect.Modifier public static final int ABSTRACT 1024 public static fin ...
- mybatis使用oracle的nulls first/nulls last
nulls first/nulls last 顾名思义,就是在检索结果集里,有null值的时候,把null值认为是最大值,还是最小值. nulls first 放置在结果集最前面 nulls last ...