算法–递归–汉诺塔
提示:递归思想值汉诺塔编程转化
前言
递归思想解决汉诺塔问题编程思路解析
一、递归是什么?
借用百度文库:程序调用自身的编程技巧称为递归( recursion)。递归做为一种算法在程序设计语言中广泛应用。 一个过程或函数在其定义或说明中有直接或间接调用自身的一种方法,它通常把一个大型复杂的问题层层转化为一个与原问题相似的规模较小的问题来求解,递归策略只需少量的程序就可描述出解题过程所需要的多次重复计算,大大地减少了程序的代码量。递归的能力在于用有限的语句来定义对象的无限集合。一般来说,递归需要有边界条件、递归前进段和递归返回段。当边界条件不满足时,递归前进;当边界条件满足时,递归返回。
二、汉诺塔问题
1.汉诺塔
相传在古印度圣庙中,有一种被称为汉诺塔(Hanoi)的游戏。该游戏是在一块铜板装置上,有三根杆(编号A、B、C),在A杆自下而上、由大到小按顺序放置64个金盘(如下图)。游戏的目标:把A杆上的金盘全部移到C杆上,并仍保持原有顺序叠好。操作规则:每次只能移动一个盘子,并且在移动过程中三根杆上都始终保持大盘在下,小盘在上,操作过程中盘子可以置于A、B、C任一杆上。
分析:对于这样一个问题,任何人都不可能直接写出移动盘子的每一步,但我们可以利用下面的方法来解决。设移动盘子数为n,为了将这n个盘子从A杆移动到C杆,可以做以下三步:
(1)以C盘为中介,从A杆将1至n-1号盘移至B杆;
(2)将A杆中剩下的第n号盘移至C杆;
(3)以A杆为中介;从B杆将1至n-1号盘移至C杆。 [2]
这样问题解决了,但实际操作中,只有第二步可直接完成,而第一、三步又成为移动的新问题。以上操作的实质是把移动n个盘子的问题转化为移动n-1个盘,那一、三步如何解决?事实上,上述方法设盘子数为n, n可为任意数,该法同样适用于移动n-1个盘。因此,依据上法,可解决n -1个盘子从A杆移到B杆(第一步)或从B杆移到C杆(第三步)问题。现在,问题由移动n个盘子的操作转化为移动n-2个盘子的操作。依据该原理,层层递推,即可将原问题转化为解决移动n -2、n -3… … 3、2,直到移动1个盘的操作,而移动一个盘的操作是可以直接完成的。至此,我们的任务算作是真正完成了。而这种由繁化简,用简单的问题和已知的操作运算来解决复杂问题的方法,就是递归法。在计算机设计语言中,用递归法编写的程序就是递归程序。
2.汉诺塔理解
- 一个盘子
过程如图:
- 两个盘子
- 三个盘子
- 总结
由上图可以看出都可对应上文三步:
(1)把n-1个盘子A移动到B
(2)把第n个盘子A移动到B
(3)把n-1个盘子B移动到C
拆解为树状结构如下图所示,每一次拆解都可拆成三层结构。
转换为代码(C#)
递归代码:
/// <summary>
/// 汉诺塔函数 n 表示几个盘子 ,x,y,z分别代表三个柱子
/// </summary>
/// <param name="n"></param>
/// <param name="x"></param>
/// <param name="y"></param>
/// <param name="z"></param>
public void hanoi(int n, char x, char y, char z)
{
if (n == 1)
{
move(x, z); //一个柱子时候:从第x移动到第z柱子
}
else
{
hanoi(n - 1, x, z, y);//从第n-1柱子从x移动到y柱子
move( x, z); //从第n柱子从x移动到z柱子
hanoi(n - 1, y, x, z); //从第n-1柱子从y移动到z柱子
}
}
移动代码
/// <summary>
/// 移动函数 将从From移动到To处
/// </summary>
/// <param name="From"></param>
/// <param name="To"></param>
public void move( char From, char To)
{
Console.WriteLine("从" + From + "移动到" + To);
}
主函数:
static void Main(string[] args)
{
//using下面 创建类NumberManipulator
NumberManipulator n = new NumberManipulator();
n.hanoi(4,'A','B','C');
Console.ReadLine();
}
关键点注释:
不论n为多少,其最终都要去执行move(x,z)
函数,要需保证其循环第一步为从x移动到y就要将hanoi(n,x,y,z)
中n-1 同时将y与z互换,互换后对n-1执行move(x,z)
时便是将n-1个盘子由x移动到y上;同理将n-1个盘子从y移动到z上。
附上我手写的流程
总结
递归思想理解起来还相对容易,关键是转换为程序时理解,我是从n=1-4写了下流程后才明白些的,关键是我在C#中单步调试看程序怎样运行也还是有点晕,最后写了写才算勉强明白吧,写成博文给大家分享出来,希望有所帮助。