题目描述

  有 \(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】取石子 博弈论的更多相关文章

  1. 【ACM】取石子 - 博弈论

    取石子(一) 时间限制:3000 ms  |  内存限制:65535 KB 难度:2   描述 一天,TT在寝室闲着无聊,和同寝的人玩起了取石子游戏,而由于条件有限,他/她们是用旺仔小馒头当作石子.游 ...

  2. bzoj 3895 取石子——博弈论

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3895 看题解:https://blog.csdn.net/popoqqq/article/d ...

  3. bzoj 3895 取石子 —— 博弈论

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3895 看了博客:https://blog.csdn.net/popoqqq/article/ ...

  4. 【BZOJ1413】[ZJOI2009]取石子游戏(博弈论,动态规划)

    [BZOJ1413][ZJOI2009]取石子游戏(博弈论,动态规划) 题面 BZOJ 洛谷 题解 神仙题.jpg.\(ZJOI\)是真的神仙. 发现\(SG\)函数等东西完全找不到规律,无奈只能翻题 ...

  5. POJ.1067 取石子游戏 (博弈论 威佐夫博弈)

    POJ.1067 取石子游戏 (博弈论 威佐夫博弈) 题意分析 简单的威佐夫博弈 博弈论快速入门 代码总览 #include <cstdio> #include <cmath> ...

  6. HDU.2516 取石子游戏 (博弈论 斐波那契博弈)

    HDU.2516 取石子游戏 (博弈论 斐波那契博弈) 题意分析 简单的斐波那契博弈 博弈论快速入门 代码总览 #include <bits/stdc++.h> #define nmax ...

  7. bzoj3895: 取石子(博弈论,记忆化搜索)

    3895: 取石子 Time Limit: 1 Sec  Memory Limit: 512 MBSubmit: 361  Solved: 177[Submit][Status][Discuss] D ...

  8. 取石子游戏 HDU 1527 博弈论 威佐夫博弈

    取石子游戏 HDU 1527 博弈论 威佐夫博弈 题意 有两堆石子,数量任意,可以不同.游戏开始由两个人轮流取石子.游戏规定,每次有两种不同的取法,一是可以在任意的一堆中取走任意多的石子:二是可以在两 ...

  9. HDU 2516 取石子游戏 (博弈论)

    取石子游戏 Problem Description 1堆石子有n个,两人轮流取.先取者第1次能够取随意多个,但不能所有取完.以后每次取的石子数不能超过上次取子数的2倍.取完者胜.先取者负输出" ...

随机推荐

  1. 【20190228】JavaScript-获取子元素

    在写JavaScript的时候发现了一个获取子节点的坑,如以下的html结构 <div id="parent"> <div>1</div> &l ...

  2. vivo如何录制手机视频 分享简单的操作方法

    智能手机功能不断的发展更新,手机已经普及到每一个人,在日常的生活或者工作中都离不开手机,手机中的功能例如一些小视频软件都是非常有趣的,vivo如何录制手机视频?下面我们一起来看看吧! 使用工具:手机 ...

  3. 理解java的三种代理模式

    代理模式是什么 代理模式是一种设计模式,简单说即是在不改变源码的情况下,实现对目标对象的功能扩展. 比如有个歌手对象叫Singer,这个对象有一个唱歌方法叫sing(). 1 public class ...

  4. Nginx+uWSGI启动Django

    在之前的几篇博客中对Django的功能做了初步实践,这里链接贴一下: Django的安装和启动 Django之--网页展示Hello World! Django之--通过MVC架构的html模板展示H ...

  5. GDB 显示别的文件;在别文件打断点;执行到函数末尾;跳出当前函数

    显示别的文件:l "文件名.后缀名":行号 在别文件打断点:b "文件名.后缀名":行号 执行到函数末尾:finish 跳出当前函数(当前位置到函数的末尾不被执 ...

  6. 如何提高 windows 的使用效率?--巧用运行命令

    windows 操作系统可以使用 win+R 运行一些命令执行任务,好处是:高效.快速.准确. 启动程序 将程序 chrome 写入以下注册表中, SOFTWARE\Microsoft\Windows ...

  7. iOS Accessibility指南

    开发者经常会为用户开发一些令人充满惊喜的App.但是,开发者真的为每一个潜在的用户都做适配了么?是否每个人都可以真正使用你的APP呢? 设计APP.产品或者任何类型的服务,都要考虑到所有用户,包括视力 ...

  8. maven-assembly-plugin打包可执行的jar包

    pom.xml添加 <build> <plugins> <plugin> <artifactId>maven-assembly-plugin</a ...

  9. 平滑升级你的Nginx

    1.概述(可以直接跳过看第2部分) Nginx方便地帮助我们实现了平滑升级.其原理简单概括,就是: (1)在不停掉老进程的情况下,启动新进程. (2)老进程负责处理仍然没有处理完的请求,但不再接受处理 ...

  10. Java 8 新特性:6-Optional类

    (原) 先看看上面的说明: /** * A container object which may or may not contain a non-null value. * If a value i ...