题目描述

两人进行 $T$ 轮游戏,给定参数 $F$ ,每轮给出 $N$ 堆石子,先手和后手轮流选择石子数大于等于 $F$ 的一堆,将其分成任意(大于1)堆,使得这些堆中石子数最多的和最少的相差不超过1(即尽量均分)。求先手和后手谁必胜。

输入

输入第一行包含两个正整数T和F,分别表示游戏组数与给定的数。
接下来T行,每行第一个数N表示该组游戏初始状态下有多少堆石子。之后N个正整数,表示这N堆石子分别有多少个。

输出

输出一行,包含T个用空格隔开的0或1的数,其中0代表此时小A(后手)会胜利,而1代表小A的对手(先手)会胜利。

样例输入

4 3 
1 1 
1 2 
1 3 
1 5

样例输出

0 0 1 1


题解

博弈论+SG定理+数学

显然游戏为多状态组合游戏,考虑使用SG定理求解。

SG定理:定义每种局面的SG函数值为该局面所有子局面SG函数值的mex,多状态组合游戏的SG值为每个状态SG值的异或。SG值为0时先手必败,否则先手必胜。

设 $f[i]$ 表示一堆 $i$ 个石子的SG函数值,当 $i<f$ 时 $f[i]=0$ 。枚举分成的堆数 $j$ ,根据定义有 $f[i]=\text{mex}_{j=2}^{i}f[\frac ij]\text{^}f[\frac ij]\text{^}...\text{^}f[\frac ij]\text{^}f[\frac ij+1]\text{^}f[\frac ij+1]\text{^}...\text{^}f[\frac ij+1]$ 。
其中 $\text{^}$ 表示异或,$f[\frac ij]$ 有 $j-i\%j$ 个,$f[\frac ij+1]$ 有 $i\%j$ 个,$\%$ 表示模。

这里出现了 $\frac ij$ ,可以使用数学下底分块的方法枚举商,求出 $\frac ij$ 相等的一段区间。

由于 $x\text{^}x=0$ ,因此只需要考虑 $j-i\%j$ 和 $i\%j$ 的奇偶性即可。

又因为 $i\%j=i-j·\frac ij$ ,因此当 $\frac ij$ 为定值时,相同奇偶性的 $j$ 对应的答案相同。

具体来讲 ,$\text{mex}$ 后面的部分当 $j$ 为偶数时:$i$ 为奇数时为 $f[\frac ij]\text{^}f[\frac ij+1]$ ,否则为 $0$ ;当 $j$ 为奇数时:$i-j$ 为奇数为 $f[\frac ij+1]$ ,否则为 $f[\frac ij]$ 。

之后暴力求 $\text{mex}$ 即可(复杂度是对的)。

最后对于询问,将每堆的SG值异或即可得到最终的SG值。

时间复杂度 $O(n\sqrt n)$ 。

bzoj上比较卡常,需要使用时间戳优化来记录每个SG是否作为子状态出现。

#include <cstdio>
#include <algorithm>
using namespace std;
int f[100010] , v[100010];
inline void calc(int n , int c)
{
int i , t , last;
for(i = 2 ; i <= n ; i = last + 1)
{
t = n / i , last = n / t;
if(last / 2 > (i - 1) / 2)
{
if(n & 1) v[f[t] ^ f[t + 1]] = c;
else v[0] = c;
}
if((last + 1) / 2 > i / 2)
{
if((n - t) & 1) v[f[t + 1]] = c;
else v[f[t]] = c;
}
}
}
int main()
{
int T , c , i , n , x , val;
scanf("%d%d" , &T , &c);
for(i = c ; i < 100000 ; i ++ )
{
calc(i , i);
while(v[f[i]] == i) f[i] ++ ;
}
while(T -- )
{
scanf("%d" , &n) , val = 0;
for(i = 1 ; i <= n ; i ++ ) scanf("%d" , &x) , val ^= f[x];
printf("%d" , (bool)val);
if(T) printf(" ");
}
return 0;
}

【bzoj3576】[Hnoi2014]江南乐 博弈论+SG定理+数学的更多相关文章

  1. bzoj3576: [Hnoi2014]江南乐

    Description 小A是一个名副其实的狂热的回合制游戏玩家.在获得了许多回合制游戏的世界级奖项之后,小A有一天突然想起了他小时候在江南玩过的一个回合制游戏.    游戏的规则是这样的,首先给定一 ...

  2. [HNOI2014]江南乐 博弈论

    题面 题面 题解 首先我们知道一个关于除法的重要性质:对于一个固定的\(i\),表达式\(\frac{i}{m}\)的取值只有根号个. 因此我们考虑如何优化SG函数的求解. 观察到在取值相同的同一段中 ...

  3. bzoj 3576[Hnoi2014]江南乐 sg函数+分块预处理

    3576: [Hnoi2014]江南乐 Time Limit: 30 Sec  Memory Limit: 512 MBSubmit: 1929  Solved: 686[Submit][Status ...

  4. 【BZOJ3576】江南乐(博弈论)

    [BZOJ3576]江南乐(博弈论) 题面 BZOJ 洛谷 题解 无论一堆石头怎么拆分,都并不能改变它是一个\(Multi-SG\)的事实. 既然每一组的\(F\)都是固定的,那么我们预处理所有的可能 ...

  5. 洛谷 P3235 [HNOI2014]江南乐 解题报告

    P3235 [HNOI2014]江南乐 Description 两人进行 T 轮游戏,给定参数 F ,每轮给出 N 堆石子,先手和后手轮流选择石子数大于等于 F 的一堆,将其分成任意(大于1)堆,使得 ...

  6. 【bzoj3576】[Hnoi2014]江南乐 数论分块+博弈论

    Description 小A是一个名副其实的狂热的回合制游戏玩家.在获得了许多回合制游戏的世界级奖项之后,小A有一天突然想起了他小时候在江南玩过的一个回合制游戏. 游戏的规则是这样的,首先给定一个数F ...

  7. luogu P3235 [HNOI2014]江南乐

    传送门 这题又是我什么时候做的(挠头) 首先是个和SG函数有关的博弈论,SG=0则先手必败.显然一堆石子就是一个游戏,而若干堆石子的SG值就是每堆SG的异或和,所以算出每堆石子SG就能知道答案 然后怎 ...

  8. P3235 [HNOI2014]江南乐

    $ \color{#0066ff}{ 题目描述 }$ 小A是一个名副其实的狂热的回合制游戏玩家.在获得了许多回合制游戏的世界级奖项之后,小A有一天突然想起了他小时候在江南玩过的一个回合制游戏. 游戏的 ...

  9. 【bzoj3576】 Hnoi2014—江南乐

    http://www.lydsy.com/JudgeOnline/problem.php?id=3576 (题目链接) 题意 给出一个数$F$,然后$n$堆石子,每次操作可以把一堆不少于$F$的石子分 ...

随机推荐

  1. 20155321 2016-2017-2 《Java程序设计》第一周学习总结

    学习目标 [√]了解Java基础知识 [√]了解JVM.JRE与JDK,并下载.安装.测试JDK [√]了解PATH.CLASSPATH.SOURCEPATH的作用并会设置 [√]初步使用IDE(推荐 ...

  2. python3出现转码问题的总结

    对于此(类)问题: (1)出现UnicodeEncodeError –> 说明是Unicode编码时候的问题: (2) ‘gbk’ codec can’t encode character –& ...

  3. Popup 解决置顶显示问题

    原文:Popup 解决置顶显示问题 前言 Popup显示时会置顶显示.尤其是 Popup设置了StayOpen=true时,会一直置顶显示,问题更明显. 置顶显示问题现象: 解决方案 怎么解决问题? ...

  4. BZOJ2039_employ人员雇佣_KEY

    题目传送门 网络流,求最小割. 设tot为所有盈利的和,即所有人(不花钱)雇佣. 对于S->i建一条容量为c[i]的边,i->j建一条S[i][j]*2的边,之所以这样建是因为如果不选这个 ...

  5. jQuery Validate (登录页面相关验证)

    $(function() { var submit = false; var superHtml = []; /** * 匹配企业帐号,以字母开头,长度在6-20之间,只能包含字符.数字和下划线. * ...

  6. Spring学习(十二)-----Spring @PostConstruct和@PreDestroy实例

    实现 初始化方法和销毁方法3种方式: 实现标识接口 InitializingBean,DisposableBean(不推荐使用,耦合性太高) 设置bean属性 Init-method destroy- ...

  7. 【Python入门学习】闭包&装饰器&开放封闭原则

    1. 介绍闭包 闭包:如果在一个内部函数里,对在外部作用域的变量(不是全局作用域)进行引用,那边内部函数被称为闭包(closure) 例如:如果在一个内部函数里:func2()就是内部函数, 对在外部 ...

  8. Python 内置函数介绍

    作者博文地址:http://www.cnblogs.com/spiritman/ Python Built-in Functions

  9. java运行时内存分类

    主要有java栈(虚拟机栈), 堆 ,方法区. 线程私有: 栈: 每个方法执行的时候 都会同时创建一个栈桢 Stack Frame 用于存储  局部变量表, 操作数栈,动态链接, 方法出口等信息 线程 ...

  10. 九个很有用的php功能

    1. 函数的任意数目的参数 你可能知道PHP允许你定义一个默认参数的函数.但你可能并不知道PHP还允许你定义一个完全任意的参数的函数 下面是一个示例向你展示了默认参数的函数: 1 2 3 4 5 6 ...