初识汉诺塔问题
2015-12-02 23:00
288 查看
汉诺塔
汉诺塔(Tower of Hanoi)源于印度传说中,大梵天创造世界时造了三根金钢石柱子,其中一根柱子自底向上叠着64片黄金圆盘。大梵天命令婆罗门把圆盘从下面开始按大小顺序重新摆放在另一根柱子上。并且规定,在小圆盘上不能放大圆盘,在三根柱子之间一次只能移动一个圆盘。
——引自维基百科
若给汉诺塔传说中三根柱子分别用英文字母a,b,c命名,其中只有a柱子摆放n片圆盘(1<=n<=100000), 若要把a柱子上的所有圆盘转移到c柱子上,问最少需要移动多少次圆盘。
移动圆盘的规则如下:
每次只能移动一片圆盘
直径大的圆盘必须摆放在直径小的圆盘之上
递归求解
汉诺塔问题通过简单的递归进行求解,代码比较简洁,通俗易懂。其实汉诺塔问题的移动次数是有规律可寻的,通过递归代码找出相应的规律,并通过数学方法得到结果效率才是最高的。当n=1时,a柱子只有一个圆盘,直接移至c柱
当n>1时,根据规则1和2,将a柱子n-1个圆盘移动到b柱子,然后将a剩下的一个圆盘移动到c,接着再把b上暂时放着的n-1个圆盘移动到c
递归求解其实就是不断降低问题规模的过程,将b柱子的n-1个圆盘移至c何尝不重复上述两点的过程。递归求解的C代码如图3-1所示。
void Hanoi(int n, char a, char b, char c) { if(n == 1) { Move(a, c); } else { Hanoi(n-1, a, c, b); /*将a柱子n-1个圆盘移动到b柱子*/ Move(a, c); /*将a剩下的一个圆盘移动到c*/ Hanoi(n-1, b, a, c); /*再把b上暂时放着的n-1个圆盘移动到c*/ } } void Move(char a, char b) { printf("Move 1 disk: %c ---------> %c\n", a, b); }
图3-1 汉诺塔递归求解代码
如图3-1的代码可以得出汉诺塔移动圆盘次数的递推关系:
Hanoi(n) = 1 , n =1 ;
Hanoi(n) = 2 * Hanoi(n-1) + 1, n>1;
令b(n) = Hanoi(n)+1,可以得出b(n)是一个以2为底,公比为2的等比数列。由此可得,
b(n) = 2n, n>0
Hanoi(n) = 2n - 1, n>0
增加约束条件的汉诺塔
Q: 若加上一个限制条件,圆盘只能在相邻柱子之间移动,又如何解决?假设a, b, c并排,b在中间,即a, c不相邻,把a上的一个圆盘移动到c上必须先移至b,然后再移动到c。
From TOJ 3270 Strange Hanoi Tower
同理使用递归求解,根据原有规则和新增约束条件,推导出移动圆盘的次数的规律。
当n=1时,a柱子只有一个圆盘,先移到b,再移至c
当n>1时,先将a柱的n-1个圆盘通过b柱移至c柱,再将a柱子剩下的一个圆盘移到b; 接着把c柱n-1个圆盘通过b柱移到a; 然后把b柱子目前唯一的圆盘移至c,最后把a柱子的n-1个圆盘通过b移至c
递归代码如图3-2所示。
void Hanoi(int n, char a, char b, char c) { if(n==1) { //a柱子只有一个圆盘,先移到b,再移至c Move(a, b); Move(b, c); } else { Hanoi(n-1, a, b, c); //先将a柱的n-1个圆盘通过b柱移至c柱 Move(a, b); //a柱子剩下的一个圆盘移到b Hanoi(n-1, c, b, a); //把c柱n-1个圆盘通过b柱移到a Move(b, c); //把b柱子目前唯一的圆盘移至c Hanoi(n-1, a, b ,c); //把a柱子的n-1个圆盘通过b移至c } } void Move(char a, char b) { printf("%c --> %c\n", a, b); }
图3-2 增加约束条件的汉诺塔递归代码
由图3-2可得,移动圆盘次数与圆盘个数的递推关系,
当n=1时,Hanoi(n) = 2;
当n>1时,Hanoi(n) = 3 * Hanoi(n-1) + 2;
最后求得:
Hanoi(n) = 3n - 1, n>0
这道题目的结果可能会很大,输出的结果要 mod 1,000,000,007。
高阶整数幂取模
因此TOJ 3270这道题目转化为求 (3n - 1) % 1000000007的数学问题。对高阶整数幂取模运算,可以通过除幂处理。题目的代码如图3-3所示。#include <stdio.h> const long long p = 1000000007; long long power(long long x, long long n) {// if( n == 0) return 1; else if(n == 1) return x; else { long long tmp = power(x, n/2); if( n % 2 == 1) return tmp * tmp * x % p; // 若 n 为奇,3n = 3 * 3n/2 * 3n/2 else return tmp * tmp % p; // 若 n 为偶,3n = 3n/2 * 3n/2 } } int main() { long long n; while(scanf("%lld", &n) && n) { printf("%lld\n", power(3, n)-1); } return 0; }
图3-3 TOJ 3270 Strange Hanoi Tower的参考代码
思考:为什么得到的结果要取模?
任何一个质数总能除尽任何几何级数中的某一项减1,且该项的指数是这个给定的质数减1的因子。
—— 费马
有兴趣的读者可以阅读神奇的费马小定理这篇文章。
相关文章推荐
- 利用Theano理解深度学习——Multilayer Perceptron
- 【iOS】libc++abi.dylib: terminate_handler unexpectedly threw an exception
- Android Studio 使用之一安装设置
- vector调用erase 后运行时异常vector iterators incompatible
- 【转】SVN提交一般原则
- theano中的scan用法
- 搭建redmine--RVM
- Java 线程学习
- hibernate3和struts2和spring使用到的jar包详解
- Qtcreator常用快捷键
- 剑指offer系列之一:从尾到头打印链表
- conprovider的监听
- 利用反射完成初级万能DAO
- CentOS Linux解决Device eth0 does not seem to be present
- V4L2规范编程--21
- JAVA_SE ----- 基础知识总结-----运算符,键盘录入,语句
- 原理图初识之GPIO
- lightoj1123 Trail Maintenance
- oracle中的检查点Ckpt和SCN号
- 1202实验三 进程调度实验