Fibonacci again and again HDU - 1848(尼姆博弈+SG函数的运用+SG函数详解)

题意:

给出三堆石子(m,n,p个),两人每次只能取斐波那契数f[i]个,最先取光所有石子者取胜

题目:

任何一个大学生对菲波那契数列(Fibonacci numbers)应该都不会陌生,它是这样定义的:
F(1)=1;
F(2)=2;
F(n)=F(n-1)+F(n-2)(n>=3);
所以,1,2,3,5,8,13……就是菲波那契数列。
在HDOJ上有不少相关的题目,比如1005 Fibonacci again就是曾经的浙江省赛题。
今天,又一个关于Fibonacci的题目出现了,它是一个小游戏,定义如下:
1、 这是一个二人游戏;
2、 一共有3堆石子,数量分别是m, n, p个;
3、 两人轮流走;
4、 每走一步可以选择任意一堆石子,然后取走f个;
5、 f只能是菲波那契数列中的元素(即每次只能取1,2,3,5,8…等数量);
6、 最先取光所有石子的人为胜者;

假设双方都使用最优策略,请判断先手的人会赢还是后手的人会赢。

Input

输入数据包含多个测试用例,每个测试用例占一行,包含3个整数m,n,p(1<=m,n,p<=1000)。
m=n=p=0则表示输入结束。

Output

如果先手的人能赢,请输出“Fibo”,否则请输出“Nacci”,每个实例的输出占一行。

Sample Input

1 1 1
1 4 1
0 0 0

Sample Output

Fibo
Nacci

分析:

SG函数解组合游戏
SG定理: 游戏“和”的SG函数等于各子游戏SG函数的Nim和。

所谓的SG值就是记录当前状态是N是P的具体值,N-position表示必赢状态(其SG值不为0),P-position表示必输状态(其SG值为0)。
下面介绍怎么求SG值:首先定义mex(minimal excludant)运算,这是施加于一个集合的运算,表示不属于mex这个集合的最小非负整数。例如mex{0,1,2,4}=3、mex{2,3,5}=0、mex{}=0。对于一个给定的有向无环图,定义关于图的每个顶点的Sprague-Grundy函数g如下:
g(n)=mex{ g(m) | m是n的后继 },这里的g(n)即dp[n]
拿本题的栗子来讲:首先有dp[0]=0,f[]={1,2,3,5…};(f数组存放可以抓走石子的个数(斐波那契数列),并且按升序存放)
举某一堆的例子。
当n=1时,先手可以抓走1-f{1}个石子,剩余{0}张,mex{dp[0]}={0},故dp[1]=1;
当n=2时,先手可以抓走2-f{1,2}个石子,剩余{1,0}个石子,mex{dp[1],dp[0]}={1,0},故dp[2]=2;
当n=3时,先手可以抓走3-f{1,2,3}个石子,剩余{2,1,0}个石子,mex{dp[2],dp[1],dp[0]}={2,1,0},故dp[3]=3;
当n=4时,先手可以抓走4-f{1,2,3}个石子,剩余{3,2,1}个石子,mex{dp[3],dp[2],dp[1]}={3,2,1},故dp[4]=0;
当n=5时,先手可以抓走5-f{1,2,3,5}个石子,剩余{4,3,2,0}个石子,mex{dp[4],dp[3],dp[2],dp[0]}={0,3,2,0},故dp[5]=1;
以此类推…
n 0 1 2 3 4 5 6 7 8 9…
dp[n] 0 1 2 3 0 1 2 3 4 5…
由上述实例我们就可以得到1~n的SG值的计算步骤,如下所示:
①、使用f数组保存菲波那契数列。
②、然后使用book数组来标记当前状态n的后继m状态。
③、最后模拟mex运算,也就是我们在集合mex中查找未被标记值的最小值,将其赋值给dp(n)。
④、不断的重复 ② - ③ 的步骤,即可完成计算1~n的SG值。

关于3种SG值计算方法(重点):
1、可选步数为1~m的连续整数,直接取模即可,SG(x) = x % (m+1);
2、可选步数为任意步,dp(x) = x;
3、可选步数为一系列不连续的数,用get_SG()计算
此题就是选取第3种方法来计算SG值。
若还不清楚,下面的代码有步骤详解:

AC代码:

#include<stdio.h>
#include<string.h>
#include<algorithm>
using namespace std;
const int M=1e3+10;
int f[M],dp[M],book[M];
int m,n,p;
/**
SG值:一个点的SG值就是一个不等于它的后继点的SG的且大于等于零的最小整数。
同mex()函数。简单点来讲就是当前状态离最近一个必败点的距离。距离为0就是必败点
SG(x)=mex(S),S是x的后继状态的SG函数值集合,mex(S)表示不在S内的最小非负整数
SG值是P/N状态的具体化
*/
int main()
{f[0]=1;f[1]=2;for(int i=2; i<100; i++)初始化f[i]=f[i-1]+f[i-2];for(int i=1; i<M; i++)///SG函数的运用;{memset(book,0,sizeof(book));///每轮到当前i就重新初始化book都为未访问状态,找出不属于这个集合的最小非负整数for(int j=0; f[j]<=i; j++)///此时j值不会越界,所以不考虑越界约束book[dp[i-f[j]]]=1;///i-f[j]为后继状态,book[sg[i-f[j]]]收录mex集合for(int j=0; j<M; j++)///求没有出现在mex集合中的非负最小值if(!book[j]){dp[i]=j;break;//每次break退出时就取不属于mex集合的最小非负整数}}while(~scanf("%d%d%d",&n,&m,&p)&&(n||m||p)){if(dp[m]^dp[n]^dp[p])///  对于每个堆的SG函数值,我们只需要异或判断结果,当dp[n]不为0时,即为N-position,此时先手必赢;printf("Fibo\n");elseprintf("Nacci\n");}return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/309813.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

[PAT乙级]1041 考试座位号

每个 PAT 考生在参加考试时都会被分配两个座位号&#xff0c;一个是试机座位&#xff0c;一个是考试座位。正常情况下&#xff0c;考生在入场时先得到试机座位号码&#xff0c;入座进入试机状态后&#xff0c;系统会显示该考生的考试座位号码&#xff0c;考试时考生需要换到考试…

sony android mp3播放器,入手一年,详细聊聊 索尼ZX505、艾利和SR15 两款安卓播放器的使用体验...

入手一年&#xff0c;详细聊聊 索尼ZX505、艾利和SR15 两款安卓播放器的使用体验2020-12-04 16:00:21113点赞179收藏28评论创作立场声明&#xff1a;本文所测商品为自费购入。如参加张大妈家的活动获得&#xff0c;我会在文中点明。坚持独立的评价观点是笔者创作的基本底线&…

研发协同平台持续交付2.0架构演进

源宝导读&#xff1a;为了打通CI/CD环节&#xff0c;实现持续的端到端的交付能力&#xff0c;RDC平台提供了在线化的更新服务&#xff0c;随着业务量增长与场景的需要&#xff0c;我们对更新服务架构重新设计&#xff0c;实现了2.0版本。本文将介绍更新服务2.0的架构演进过程与…

Aeroplane chess HDU - 4405(期望dp)

题意: 飞行棋。有n1格&#xff0c;开始时在0号格子&#xff0c;每一步都要扔一个dice&#xff08;六个面&#xff0c;概率相同&#xff09;哪一面朝上他就会向前走xi步。当xi大于等于N的时候&#xff0c;游戏结束。另外&#xff0c;地图上有m条航线。第i条航线可以直接从xi到y…

[C++11]可调用对象包装器function

可调用对象包装器 std::function是可调用对象的包装器。它是一个类模板&#xff0c;可以容纳除了类成员(函数)指针之外的所有可调用对象。通过指定它的模板参数&#xff0c;它可以用统一的方式处理函数&#xff0c;函数对象&#xff0c;函数指针&#xff0c;并允许保存和延迟执…

android应用窗口模式,[技巧]如何启用Android N开发者预览版中的“自由窗口”模式...

这里是Android N开发者预览版“自有窗口”模式的一些实际演示截图。对于已经参加了“Beta Program”的人们来说&#xff0c;通过OTA获取Android N Preview更新是最简单的。但如果你非要选择“困难模式”(命令行镜像刷新)&#xff0c;则可能让设备无法再通过OTA的方式来安装未来…

LOOPS HDU - 3853(概率dp,期望)

题意&#xff1a; 有一个R*C的方格。一个人想从&#xff08;1&#xff0c;1&#xff09;走到(r,c)。在每个格子都有三种选择&#xff0c;向下&#xff0c;向右&#xff0c;或者原地不动。每个格子里的每个选择都有一定的概率。而每次移动都需要消耗2点的能量&#xff0c;问期望…

Sql Server之旅——第四站 你必须知道的非聚集索引扫描

非聚集索引&#xff0c;这个是大家都非常熟悉的一个东西&#xff0c;有时候我们由于业务原因&#xff0c;sql写的非常复杂&#xff0c;需要join很多张表&#xff0c;然后就泪流满面了。。。这时候就有DBA或者资深的开发给你看这个猥琐的sql&#xff0c;通过执行计划一分析,或许…

[C++11]可调用对象绑定器

std::bind用来将可调用对象与其参数一起进行绑定。绑定后的结果可以使用std::function进行保存&#xff0c;并延迟调用到任何我们需要的时候。通俗来说&#xff0c;它主要有两个作用: 1.将可调用对象与其参数一起绑定成一个仿函数。 2.将多元(参数个数为n&#xff0c;n > …

电视android已停止运行是什么意思,智能电视提示应用停止运行怎么办?当贝市场三招解决...

智能电视提示应用停止运行怎么办&#xff1f;当贝市场三招解决2019年11月28日 17:53作者&#xff1a;网络编辑&#xff1a;王动分享智能电视使用久了之后,电视页面会提示我们应用停止运行,这是怎么回事?当贝小编针对这个问题,整理了一份解决教程,大家可以看看有没有什么帮助。…

重磅!2020年微软开发者大会落幕,.NET迎来新机遇!

两天前微软举行了首个线上Build大会&#xff0c;而开发者成为大会里唯一的主角。和所有技术公司一样&#xff0c;开发者对于微软来说&#xff0c;同样也越来越重要了。如同血肉相依的关系&#xff0c;谁也离不开谁。在这次大会上&#xff0c;开发者是最大的宠儿&#xff0c;成了…

番茄时间有感之关于在疫情期间我与ACM不得不说的故事

哼哼~首先声明&#xff0c;我不是来讲故事的&#xff0c;我来总结一下在疫情这段时间&#xff0c;在ACM训练过程中的自我情况的总结和反思&#xff0c;嘻嘻&#xff0c;我是一个标题党&#xff0c;如果是被标题骗进来哒&#xff0c;抱歉啦&#xff0c;有句话说的好&#xff0c;…

[PAT乙级]1042 字符统计

请编写程序&#xff0c;找出一段给定文字中出现最频繁的那个英文字母。 输入格式&#xff1a; 输入在一行中给出一个长度不超过 1000 的字符串。字符串由 ASCII 码表中任意可见字符及空格组成&#xff0c;至少包含 1 个英文字母&#xff0c;以回车结束&#xff08;回车不算在内…

获取壁纸设置背景android,【Android学习】获取Bing 15天前到明天的壁纸,并设置为背景...

//设置壁纸public void setWallpaper() {new Thread() {public void run() {try {WebImage web new WebImage(imgRealPath);Bitmap bmp web.getBitmap(MainActivity.this);Message msg new Message();msg.obj bmp;msg.what 2;mHandler.sendMessage(msg);} catch (Exceptio…

基于 abp vNext 和 .NET Core 开发博客项目 - 自定义仓储之增删改查

上一篇文章我们用Code-First的方式创建了博客所需的实体类&#xff0c;生成了数据库表&#xff0c;完成了对EF Core的封装。本篇说一下自定义仓储的实现方式&#xff0c;其实在abp框架中已经默认给我们实现了默认的通用(泛型)仓储&#xff0c;IRepository<TEntity, TKey>…

计算机操作系统第四章作业

计算机操作系统第四章作业 1.何为静态链接&#xff1f;静态链接时需要解决两个什么问题? 答&#xff1a;静态链接是指在程序运行之前&#xff0c;先将各自目标模块及它们所需的库函数&#xff0c;链接成一个完整的装入模块&#xff0c;以后不再拆开的链接方式。   将几个目…

[PAT乙级]1046 划拳

划拳是古老中国酒文化的一个有趣的组成部分。酒桌上两人划拳的方法为&#xff1a;每人口中喊出一个数字&#xff0c;同时用手比划出一个数字。如果谁比划出的数字正好等于两人喊出的数字之和&#xff0c;谁就赢了&#xff0c;输家罚一杯酒。两人同赢或两人同输则继续下一轮&…

走进WebApiClientCore的设计

WebApiClientWebApiClient是NCC开源社区的一个项目&#xff0c;是目前微服务里http接口调用的一把锋利尖刀&#xff0c;项目早期设计与开发的时候&#xff0c;是基于.netframework的&#xff0c;然后慢慢加入netstandard和netcoreapp多个框架的支持&#xff0c;设计能力出众&am…

android 片段,android – 将片段添加到片段中(嵌套片段)

我想动态地将youtube片段添加到我已经存在的片段中.我使用的代码如下&#xff1a;// setting the Youtube Player Dynamicallyprivate int setYoutubePlayer(String desc,View view,int prevID,Bundle input) {if (desc.indexOf("") ! -1) {desc desc.substring(des…

HDU 3062 Party(2-sat题模板+tarjan )

题目: 有n对夫妻被邀请参加一个聚会&#xff0c;因为场地的问题&#xff0c;每对夫妻中只有1人可以列席。在2n 个人中&#xff0c;某些人之间有着很大的矛盾&#xff08;当然夫妻之间是没有矛盾的&#xff09;&#xff0c;有矛盾的2个人是不会同时出现在聚会上的。有没有可能会…