11. Python | 递归函数_引申_汉诺塔游戏策略总结

汉诺塔游戏,是个佷益智的游戏,其中也蕴含了一些数学的知识,故事的背景甚至蕴含了一些古人对宇宙的思考。今天我们来聊下这个游戏,仅仅从游戏的方法和策略的总结上对这个游戏进行一下解剖:

游戏简介

汉诺塔_益智游戏
`由来与传说:`

法国数学家`爱德华·卢卡斯`曾编写过一个印度的古老传说:在世界中心贝拿勒斯
(在印度北部)的圣庙里,一块黄铜板上插着三根宝石针。印度教的主神`梵天`
在创造世界的时候,在其中一根针上从下到上地穿好了由大到小的64片金片,这
就是所谓的汉诺塔。不论白天黑夜,总有一个僧侣在按照下面的法则移动这些金
片:一次只移动一片,不管在哪根针上,小片必须在大片上面。僧侣们预言,当所
有的金片都从梵天穿好的那根针上移到另外一根针上时,世界就将在一声霹雳中消
灭,而`梵塔`、`庙宇`和`众生`也都将同归于尽。 

不管这个传说的可信度有多大,如果考虑一下把64片金片,由一根针上移到另一
根针上,并且始终保持上小下大的顺序。这需要多少次移动呢?这里需要递归的方
法。假设有n片,移动次数是f(n).显然f(1)=1,f(2)=3,f(3)=7,且f(k+1)=2*f(k)+1。
此后不难证明f(n)=2^n-1。n=64时,

假如每秒钟一次,共需多长时间呢?一个平年365天有31536000 秒,闰年366天有
31622400秒,平均每年31556952秒,计算一下:
                                        18446744073709551615秒

这表明移完这些金片需要5845.54亿年以上,而地球存在至今不过45亿年,太阳系
的预期寿命据说也就是数百亿年。真的过了5845.54亿年,不说太阳系和银河系,
至少地球上的一切生命,连同梵塔、庙宇等,都早已经灰飞烟灭。

游戏的目的

  • 很多人最初接触到这个益智游戏,都是直接用5个以上圆盘开始游戏的,但是又不太清楚具体的移动规律,导致玩儿了半天也没有将所有圆盘成功移动到C柱子上,甚至玩过以后就得出了一个结论,这游戏太没意思了。。。

  • 实际上这种益智类的游戏,往往都是具有一定的方法和规律可循的,因为这类游戏的目的就是让人开发思维,启迪智慧。通常这类游戏或者玩具是最适合给孩子玩的。

  • 既然这个游戏有益智的作用,我们就应该掌握玩这种游戏的方法或者至少我们应该学会教孩子如何玩这个游戏,才能让这个益智游戏开发孩子的思维启迪孩子的智慧啊!

游戏的策略

在这一部分,我们要分为两点去讲:1. 推导和归纳;2. Python代码的实现。

1. 推导和归纳

说到归纳和总结,我们可以参照数学上的归纳方法。
举个例子:

`                     1 = 1                               `
`                   1×2 = 2                               `
`                 1×2×3 = 6                               `
`               1×2×3×4 = 24                              `
`             1×2×3×4×5 = 120                             `
`                 ...                                     `
`     1×2×3×...×(n-1)×n = f(n)                            `

上面的归纳的是一个函数,也就是阶乘,表达式为:
f(n) = n! = n * (n-1) * ... * 1 = n * (n-1)!
归纳之后便形成了一个比较抽象、但是很容易记住的方法——公式。

通过归纳和演绎,用一些简单的、易操作的数字和步骤进行推演,形成适用于更广范围的通行公式。

  • 我们来看一下汉诺塔如何推演
    要求:将圆盘从 <起点A柱子>借助<缓冲B柱子>移动到<终点C柱子>
    分析:为了更好的解释说明操作步骤,我们讲圆盘从小到大依次取名为1、2、3、4、5...
    步骤:

    1. 当A柱子上有1个圆盘时
      A-->C表示将1移到C:1-->C:圆盘为奇数时,第一步指向终点C柱

    2. 当A柱子上有2个圆盘时
      ①A-->B表示将1移到B:1-->B:圆盘为偶数时,第一步指向缓冲B柱
      ②A-->C表示将2移到C:2-->C:最下面的圆盘,移动到终点C柱
      ③B-->C表示将1移到C:1-->C:缓冲柱上的圆盘,移动到终点C柱

    3. 当A柱子上有3个圆盘时
      ①A-->C表示将1移到C:1-->C:圆盘为奇数时,第一步指向终点C柱
      ①A-->B表示将2移到B:2-->B:将2号圆盘移到缓冲B柱
      ①C-->B表示将1移到B:1-->B:将1号圆盘移到缓冲B柱
      ②A-->C表示将3移到C:3-->C:最下面的圆盘,移到到终点C柱
      ③B-->A表示将1移到A:1-->A:将1号圆盘移到起点A柱
      ③B-->C表示将2移到C:2-->C:将2号圆盘移到终点C柱
      ③A-->C表示将1移到C:1-->C:将1号圆盘移到终点C柱

  • 根据上面写的3个推演步骤,我们进行归纳如下:

    1. 整个移动过程(排除只有一个圆盘的特殊情况),可以分为三个阶段,用①②③表示。
    2. ①阶段,把 n-1个圆盘移到缓冲B柱
      ②阶段,把第n个圆盘移到终点C柱
      ③阶段,把 n-1个圆盘移到终点C柱
    3. 其中,第②阶段,是在最中间的一步,第①阶段和第③阶段的移动步数是一样多的。
    4. 第一步的移动方向,根据圆盘的个数不同而定
      奇数个圆盘,第一步一定是移到终点C柱
      偶数个圆盘,第一步一定是移到缓冲B柱
      别小瞧这个方向的定位,以后每次移动1、2圆盘时,都是按照这个方向定位的。
    5. 我们把移动圆盘的过程分为n轮,每一轮都要移动1、2、1和另一个圆盘;
      并且每一轮移动1、2、1圆盘的时候,都是按照固定的规律去移动的。
      我们会以1、2圆盘同时所在的柱子为视角做定位,判断移动的方向。下面有一个图,上面详细写了具体的移动顺序和方法:
    6. 要想明白具体是如何做的,只用眼睛看是很难理解的,当你跟着我写的步骤操作几次之后,你一定会豁然开朗,原来移动起来是这么简单~

2. Python代码的实现
当然了用文字,或者截图去描述这个过程,无论如何都算不上简单明了。Python语言中,用函数将这个游戏的移动方法,定义成了一个递归函数,写法竟然非常的简单:

# 请编写 `move(n, a, b, c)`函数,它接收参数 n,表示3个柱子A、B、C中
# 第1个柱子A的盘子数量,然后打印出把所有盘子从A借助B移动到C的方法
# -*- coding: utf-8 -*-
def move(n, a, b, c):

    if n == 1:
        print(a, '-->', c)
    # 下面else的部分是需补全的代码,也就是移动方法
    else:
        move(n-1,a,c,b)   # 思考一下为什么要这样写?
        move(1,a,b,c)   # 想一下,这是哪个阶段?
        move(n-1,b,a,c)   # 你知道,这最后的一步,是什么意思吗?
# 有3个圆盘时
move(3, 'A', 'B', 'C')
# 换行
print('\n')
# 有4个圆盘时
move(4, 'A', 'B', 'C')

# 这是有3个圆盘时的【期待输出结果】:
#     A --> C
#     A --> B
#     C --> B
#     A --> C
#     B --> A
#     B --> C
#     A --> C

你可以复制一下这段代码,去验证一下结果是不是和你自己移动的结果一样!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 230,048评论 6 542
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 99,414评论 3 429
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 178,169评论 0 383
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 63,722评论 1 317
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 72,465评论 6 412
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 55,823评论 1 328
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 43,813评论 3 446
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 43,000评论 0 290
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 49,554评论 1 335
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 41,295评论 3 358
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 43,513评论 1 374
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 39,035评论 5 363
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 44,722评论 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 35,125评论 0 28
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 36,430评论 1 295
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 52,237评论 3 398
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 48,482评论 2 379

推荐阅读更多精彩内容

  • 专业考题类型管理运行工作负责人一般作业考题内容选项A选项B选项C选项D选项E选项F正确答案 变电单选GYSZ本规程...
    小白兔去钓鱼阅读 9,048评论 0 13
  • 选择题部分 1.(),只有在发生短路事故时或者在负荷电流较大时,变流器中才会有足够的二次电流作为继电保护跳闸之用。...
    skystarwuwei阅读 13,123评论 0 7
  • 选择题部分 1.()部门负责日常监督检查工作,安全巡视的同时进行消防检查,推动消防安全制度的贯彻落实。 A: 消防...
    skystarwuwei阅读 15,497评论 0 3
  • 前置文章:递归算法:www.greatytc.com/p/703069f3ba3f . 汉诺塔问题是来源于印度传...
    郎小凯阅读 786评论 0 1
  • 题目: 汉诺塔的移动可以用递归函数非常简单地实现。 请编写move(n, a, b, c)函数,它接收参数n,表示...
    快乐的杀马特阅读 469评论 0 0