博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
HDU 2516 (Fabonacci Nim) 取石子游戏
阅读量:5316 次
发布时间:2019-06-14

本文共 1375 字,大约阅读时间需要 4 分钟。

这道题的结论就是,石子的个数为斐波那契数列某一项的时候,先手必败;否则,先手必胜。

结论很简单,但是证明却不是特别容易。找了好几篇博客,发现不一样的也就两篇,但是这两篇给的证明感觉证得不清不楚的,没看太懂。

首先,证明要依赖一个邓肯多夫定理(Zeckendorf's Theorem):任何一个正整数一定能分解成若干个不重复且不相邻的斐波那契数之和。

首推维基百科上的英文证明,很严谨也能看懂,证明的过程中还用到了一条引理,但是很容易用数学归纳法证明,所以整个过程都是十分严谨的:http://en.wikipedia.org/wiki/Zeckendorf%27s_theorem

然后对于一个必胜状态,则这个数可以分解成多个斐波那契数之和,或者可以对应地将这堆石子分成若干堆来看,而且每堆石子的数量都是一个斐波那契数。

先手取完最小的那个石子堆,其数量为Fi,由于邓肯多夫定理定理,第二小的石子堆Fj ≥ Fi+2 = Fi + Fi+1 > 2Fi,所以后手面对的局面就是若干个斐波那契数,而且不能一次取完最小的石子堆。然后等后手取完以后再继续将剩下的石子数分解,再取最小的斐波那契数即可。

 

对于一个先手必败状态,也就是一个斐波那契数局面,证明不太会证。

这有一篇证明,但是感觉不严谨啊:http://blog.csdn.net/acm_cxlove/article/details/7835016

里面说

如果先手第一次取的石子数y>=f[k-1]/3,则这小堆所剩的石子数小于2y,即后手可以直接取完

这是对的,但是他为什么没有讨论y < f[k-1]/3的情况嘞?

 

然后又找到一篇证明,http://yjq24.blogbus.com/logs/46150651.html

前面还好,但是最后那个不等式我又是看得不明不白的。不知道i,j分别代表什么的下标,+_+

 

于是还是感慨一下:看论文找证明还是“帆 樯”看英文资料,里面的证明十分简洁严谨。

1 #include 
2 #include
3 using namespace std; 4 typedef long long LL; 5 6 const LL maxn = (1LL << 31); 7 LL a[50]; 8 9 int main()10 {11 a[0] = 1, a[1] = 2;12 for(int i = 2; a[i-1] <= maxn; i++)13 a[i] = a[i-1] + a[i-2];14 LL n;15 while(scanf("%lld", &n) == 1 && n)16 {17 int i = lower_bound(a, a + 45, n) - a;18 printf("%s\n", a[i] == n ? "Second win" : "First win");19 }20 21 return 0;22 }
代码君

 

转载于:https://www.cnblogs.com/AOQNRMGYXLMV/p/4418186.html

你可能感兴趣的文章
web前端面试题2017
查看>>
ELMAH——可插拔错误日志工具
查看>>
MySQL学习笔记(四)
查看>>
【Crash Course Psychology】2. Research & Experimentation笔记
查看>>
两数和
查看>>
移动设备和SharePoint 2013 - 第3部分:推送通知
查看>>
SOPC Builder中SystemID
查看>>
MySQL数据库备份工具mysqldump的使用(转)
查看>>
NTP服务器配置
查看>>
【转】OO无双的blocking/non-blocking执行时刻
查看>>
ul li剧中对齐
查看>>
关于 linux 的 limit 的设置
查看>>
HDU(4528),BFS,2013腾讯编程马拉松初赛第五场(3月25日)
查看>>
vim中文帮助教程
查看>>
MySQL基础3
查看>>
云计算数据与信息安全防护
查看>>
RxJS & Angular
查看>>
面向对象(多异常的声明与处理)
查看>>
MTK笔记
查看>>
ERROR: duplicate key value violates unique constraint "xxx"
查看>>