题目描述

  有 \(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. uiautomatorviewer 查看元素报错: Error taking device screenshot: null 原因

    使用uiautomatorviewer 查看android某些页面元素,出现错误Error obtaining UI hierarchy  Reason: Error taking device sc ...

  2. Python函数二(函数名,闭包,迭代器)之杵臼之交

    函数名的使用: 函数名可以作为值,赋值给变量. 函数名可以作为参数传参给函数. 函数名可以作为返回值. 函数名可以作为元素存储在容器里. 闭包:在嵌套函数内,使用外层局部变量(非全局变量)就是一个闭包 ...

  3. c/c++ 重载运算符 关系,下标,递增减,成员访问的重载

    重载运算符 关系,下标,递增减,成员访问的重载 为了演示关系,下标,递增减,成员访问的重载,创建了下面2个类. 1,类StrBlob重载了关系,下标运算符 2,类StrBlobPtr重载了递增,抵减, ...

  4. c/c++ 网络编程 单纯http客户端,服务器端

    网络编程 单纯http客户端,服务器端 1,http客户端 2,http服务器端 http客户端: #include <stdio.h> #include <sys/types.h& ...

  5. 调试工具gdb

    1.1 gdb符号调试器简介 gdb是一个用来调试C和C++程序的功能强大的调试器,它能在程序运行时观察程序的内部结构和内存的使用情况. gdb主要提供以下几种功能: 监视程序中变量值的变化 设置断点 ...

  6. Python Docker 查看私有仓库镜像【转】

    文章来源:python Docker 查看私有仓库镜像 pip 安装: # 首先安装epel扩展源: yum -y install epel-release # 更新完成之后,就可安装pip: yum ...

  7. 模块简介:(random)(xml,json,pickle,shelve)(time,datetime)(os,sys)(shutil)(pyYamal,configparser)(hashlib)

    Random模块: #!/usr/bin/env python #_*_encoding: utf-8_*_ import random print (random.random()) #0.6445 ...

  8. Python从入门到放弃Day01

    Py的第一天,无非是讲一些关于电脑的一些常见的基本常识,硬件之类的cpu啊.硬盘啊.显卡啊.内存条啊什么的,之后就还有一些除了windows之外的操作系统,我之前是学网络的,Readhat也学了一部分 ...

  9. Jenkins pipeline:pipeline 使用之语法详解

    一.引言 Jenkins 2.0的到来,pipline进入了视野,jenkins2.0的核心特性. 也是最适合持续交付的feature. 简单的来说,就是把Jenkins1.0版本中,Project中 ...

  10. Jetson TX2(3)opencv3 打开usb摄像头

    ubuntu2604 opencv3.4.0 https://blog.csdn.net/ultimate1212/article/details/80936175?utm_source=blogxg ...