bzoj1188分裂游戏——博弈论、SG函数

2024-02-04 11:58

本文主要是介绍bzoj1188分裂游戏——博弈论、SG函数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Description

聪聪和睿睿最近迷上了一款叫做分裂的游戏。该游戏的规则试:共有n个瓶子,标号为0,1,2…..n-1,第i个瓶子中装有p[i]颗巧克力豆,两个人轮流取豆子,每一轮每人选择3个瓶子。标号为i,j,k,并要保证i< j,j<=k且第i个瓶子中至少要有1颗巧克力豆,随后这个人从第i个瓶子中拿走一颗豆子并在j,k中各放入一粒豆子(j可能等于k)。如果轮到某人而他无法按规则取豆子,那么他将输掉比赛。胜利者可以拿走所有的巧克力豆!两人最后决定由聪聪先
取豆子,为了能够得到最终的巧克力豆,聪聪自然希望赢得比赛。他思考了一下,发现在有的情况下,先拿的人一定有办法取胜,但是他不知道对于其他情况是否有必胜策略,更不知道第一步该如何取。他决定偷偷请教聪明的你,希望你能告诉他,在给定每个瓶子中的最初豆子数后是否能让自己得到所有巧克力豆,他还希望你告诉他第一步该如何取,并且为了必胜,第一步有多少种取法?
假定 1 < n < = 21,p[i] < = 10000
Input

输入文件第一行是一个整数t表示测试数据的组数,
接下来为t组测试数据(t<=10)。
每组测试数据的第一行是瓶子的个数n,
接下来的一行有n个由空格隔开的非负整数,表示每个瓶子中的豆子数。
Output

对于每组测试数据,输出包括两行,
第一行为用一个空格两两隔开的三个整数,表示要想赢得游戏,
第一步应该选取的3个瓶子的编号i,j,k,
如果有多组符合要求的解,那么输出字典序最小的一组。
如果无论如何都无法赢得游戏,那么输出用一个空格两两隔开的三个-1。
第二行表示要想确保赢得比赛,第一步有多少种不同的取法。
Sample Input

2

4

1 0 1 5000

3

0 0 1
Sample Output

0 2 3

1

-1 -1 -1

0


博弈论部分是OI很重要的一部分分枝,而且一般思维难度比较大,很多东西我也没有彻底理解。明天我会写一篇有关简单博弈论(SG函数)的文章,今天先上道题。
这道题的难点在于如何设定函数。
题目的要求是将某一堆取出一个,然后将后面的两堆加一。将每堆看成一个独立的游戏,这个想法是错误的,因为选一堆后还会选后面的堆,也就是说每堆都会互相影响。但我们发现每一堆中的任意石子,它们都是互相独立的,取任意一个都不会影响其它的。所以我们以石子来设函数。
我们设 sg[i] s g [ i ] 表示移动第 i i 堆里的任意一个石子,先手的胜负情况。
显然,第n堆石子是不能选的,因为它后面没有石子,也就是没有后继状况,所以 sg[n]=0 s g [ n ] = 0 ,而对于任意的 k,k[1,n) k , k ∈ [ 1 , n ) ,它的后继可以选 jkn>=jk>k j 、 k ( n >= j 、 k > k ) 根据 multi m u l t i _ SG S G 的性质,它们的游戏的和就是 sg[j] s g [ j ] ^ sg[k] s g [ k ] ,所以 sg[i]=mex(sg[j] s g [ i ] = m e x ( s g [ j ] ^ sg[k]) s g [ k ] )
这样我们就求出了 sg s g ,那么我们还要求初始的游戏的和,所以我们直接将所有石子的 sg s g 值异或起来即可。也就相当于堆 i i <script type="math/tex" id="MathJax-Element-684">i</script>如果有奇数个石子,那就异或上,否则不操作即可。
#include<bits/stdc++.h>
using namespace std;
int read(){char c;int x;while(c=getchar(),c<'0'||c>'9');x=c-'0';while(c=getchar(),c>='0'&&c<='9') x=x*10+c-'0';return x;
}
int T,n,sgtot,ans,a[24],vis[55],sg[24];
void getsg(){memset(vis,0,sizeof(vis));sg[n]=0;for(int i=n-1;i>=1;i--){for(int j=i+1;j<=n;j++)for(int k=j;k<=n;k++) vis[sg[j]^sg[k]]=i;for(int j=0;;j++)if(vis[j]!=i){sg[i]=j;break;}}
}
int main()
{T=read();while(T--){n=read();for(int i=1;i<=n;i++) a[i]=read();getsg();sgtot=0;ans=0;for(int i=1;i<=n;i++) if(a[i]&1) sgtot^=sg[i];for(int i=1;i<=n;i++)for(int j=i+1;j<=n;j++)for(int k=j;k<=n;k++)if((sgtot^sg[i]^sg[j]^sg[k])==0){ans++;if(ans==1) printf("%d %d %d\n",i-1,j-1,k-1);}if(ans==0) puts("-1 -1 -1");printf("%d\n",ans);}return 0;
}

这篇关于bzoj1188分裂游戏——博弈论、SG函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/677413

相关文章

一文详解Python如何开发游戏

《一文详解Python如何开发游戏》Python是一种非常流行的编程语言,也可以用来开发游戏模组,:本文主要介绍Python如何开发游戏的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录一、python简介二、Python 开发 2D 游戏的优劣势优势缺点三、Python 开发 3D

Python函数作用域与闭包举例深度解析

《Python函数作用域与闭包举例深度解析》Python函数的作用域规则和闭包是编程中的关键概念,它们决定了变量的访问和生命周期,:本文主要介绍Python函数作用域与闭包的相关资料,文中通过代码... 目录1. 基础作用域访问示例1:访问全局变量示例2:访问外层函数变量2. 闭包基础示例3:简单闭包示例4

Python中isinstance()函数原理解释及详细用法示例

《Python中isinstance()函数原理解释及详细用法示例》isinstance()是Python内置的一个非常有用的函数,用于检查一个对象是否属于指定的类型或类型元组中的某一个类型,它是Py... 目录python中isinstance()函数原理解释及详细用法指南一、isinstance()函数

python中的高阶函数示例详解

《python中的高阶函数示例详解》在Python中,高阶函数是指接受函数作为参数或返回函数作为结果的函数,下面:本文主要介绍python中高阶函数的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录1.定义2.map函数3.filter函数4.reduce函数5.sorted函数6.自定义高阶函数

Python中的sort方法、sorted函数与lambda表达式及用法详解

《Python中的sort方法、sorted函数与lambda表达式及用法详解》文章对比了Python中list.sort()与sorted()函数的区别,指出sort()原地排序返回None,sor... 目录1. sort()方法1.1 sort()方法1.2 基本语法和参数A. reverse参数B.

Python函数的基本用法、返回值特性、全局变量修改及异常处理技巧

《Python函数的基本用法、返回值特性、全局变量修改及异常处理技巧》本文将通过实际代码示例,深入讲解Python函数的基本用法、返回值特性、全局变量修改以及异常处理技巧,感兴趣的朋友跟随小编一起看看... 目录一、python函数定义与调用1.1 基本函数定义1.2 函数调用二、函数返回值详解2.1 有返

Python Excel 通用筛选函数的实现

《PythonExcel通用筛选函数的实现》本文主要介绍了PythonExcel通用筛选函数的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着... 目录案例目的示例数据假定数据来源是字典优化:通用CSV数据处理函数使用说明使用示例注意事项案例目的第一

C++统计函数执行时间的最佳实践

《C++统计函数执行时间的最佳实践》在软件开发过程中,性能分析是优化程序的重要环节,了解函数的执行时间分布对于识别性能瓶颈至关重要,本文将分享一个C++函数执行时间统计工具,希望对大家有所帮助... 目录前言工具特性核心设计1. 数据结构设计2. 单例模式管理器3. RAII自动计时使用方法基本用法高级用法

GO语言中函数命名返回值的使用

《GO语言中函数命名返回值的使用》在Go语言中,函数可以为其返回值指定名称,这被称为命名返回值或命名返回参数,这种特性可以使代码更清晰,特别是在返回多个值时,感兴趣的可以了解一下... 目录基本语法函数命名返回特点代码示例命名特点基本语法func functionName(parameters) (nam

Python Counter 函数使用案例

《PythonCounter函数使用案例》Counter是collections模块中的一个类,专门用于对可迭代对象中的元素进行计数,接下来通过本文给大家介绍PythonCounter函数使用案例... 目录一、Counter函数概述二、基本使用案例(一)列表元素计数(二)字符串字符计数(三)元组计数三、C