汉诺塔问题的python实现

问题是源于印度一个古老传说的益智玩具。大梵天创造世界的时候做了三根金刚石柱子,在一根柱子上从下往上按照大小顺序摞着64片黄金圆盘。大梵天命令婆罗门把圆盘从下面开始按大小顺序重新摆放在另一根柱子上。并且规定,在小圆盘上不能放大圆盘,在三根柱子之间一次只能移动一个圆盘。

代码实现:

# -*- coding: utf-8 -*-
"""
Created on Mon Sep 17 10:24:41 2018

@author: 无止境
function:汉诺塔问题
"""
step=0

def hanoi(n,a,b,c):#形参接收实参的值
    global step#修改全局变量时,用global声明,只引用不修改的话不需要global声明
    if n==1:
        print("%c-->%c\n"%(a,c))#注意python的输出
        step=step+1
    else:
        hanoi(n-1,a,c,b)#n-1个圆盘通过c移动到b
        print("%c-->%c\n"%(a,c))#把a上的一个圆盘移动到c
        step=step+1
        hanoi(n-1,b,a,c)#把b上的n-1个圆盘通过a移动到c
        
n=input("Please input n :")#获得输入
n=int(n)#转换为数值型
hanoi(n,'A','B','C')#传入字符型实参
print("Step number :",step)
#
#def move(n,a,b,c):
#    #a,b,c分别是三根柱子,n为套在a柱上的圆圈个数
#    if n==1:
#        print(a,'-->',c)
#        return
#    move(n-1,a,c,b)
#    move(1,a,b,c)
#    move(n-1,b,a,c)
#    
#move(3,'A','B','C')  
##这里我用大写字母ABC表示实参是为了与形参小写的abc区别开,这步很关键

汉诺塔问题是一个经典的递归问题,对于这个问题,我们可以把它简单的去看成是如何用n-1去表示n。

在A,B,C三个柱子上,我们先假设A柱上只有两个盘子,那么很简单,只需要把最上面的那个盘子移到B柱上,再把A柱上最下面的盘子移到C柱上,最后把B柱的盘子移到C柱就可以了。

假设我们有n个盘子,那么可以把最下面的盘子看成是第n个盘子,而我们要做的是把上面n-1个盘子移到B柱上,再把第n个盘子移到C柱。我们可以把B柱视为主中转站。

在将n-1个盘子移到B柱的过程中,我们需要借助C柱作为分中转站,当完成n-1个盘子的移动时,此时B柱上存在n-1个盘子,而我们接下来要做的,和之前类似,就是借助把n-2个盘子移动到A柱,把第n-1个盘子移动到C柱。在移动n-2个盘子到A柱时,我们同样要借助C作为分中转站。

现在问题就很清晰了,在移动的整个过程中,变换的只有AB这两个主中转站,而分中转站C是不变的,因为C作为中转的作用仅仅是为了盘子在AB之间的移动。

这样,我们很容易就可以用递归写出整个过程的代码。

猜你喜欢

转载自blog.csdn.net/xjp_xujiping/article/details/82745149