【C语言】汉诺塔问题--递归实现,有数学推导

1.汉诺塔规则介绍

在这里插入图片描述
问题如下:(汉诺塔背景放后面)
三根柱子,分别为 A,B,C。起初,有 n 个盘子从大到小全部摞在 A 柱子上
现在将其一个一个全搬到 C 柱子上,且搬运过程中盘子必须遵循由小到大的顺序,借助B 柱子,实现这一过程

在这里插入图片描述

举例:
1 个盘子: 1次
A–>C

2 个盘子: 3次
A–>B
A–>C
B–>C

3 个盘子:7次
A --> C
A --> B
C --> B
A --> C
B --> A
B --> C
A --> C

4 个盘子:15次
A --> B
A --> C
B --> C
A --> B
C --> A
C --> B
A --> B
A --> C
B --> C
B --> A
C --> A
B --> C
A --> B
A --> C
B --> C

通过举例,了解下规则

2.汉诺塔问题的数学推导

通过上面举例可以看出:
A–>C永远是中间一步
而且它上面和下面的移动过程是对称的

在这里插入图片描述

所以,移动盘子过程可分为三大步:
1.A 柱上最后一个盘子不动,上面 n-1 个盘子借助 C 全移动到 B 上
2.A–>C
3.把B上的n-1个盘子通过 A 全部移动到 C 上

上面的图很容易看出前后过程是对称的,所以移动次数必然是奇数
—————————————————————————————————
推导 n 个盘子搬运次数的通项公式:
捋逻辑:假设第n 个(最大个那个盘子)不动,只移动上面n-1个
最终结果如下图:

在这里插入图片描述
**1.**就相当于移动完n-1个盘子以后,左边又多出来一个,此时移动次数为an-1
**2.**转换目标柱子:将中间柱子定为目标柱(全移动到它上面),首先要把最左边的先移动 1 下到中间去
**3.**把右边n-1个盘子都移到中间去,相当于回到了初始移动的情景,所以次数也是an-1

写通项:(这编辑器不支持下标。。。。我用world打的)

在这里插入图片描述

3.汉诺塔问题在c语言中的递归逻辑

#include<stdio.h>
void Print_Move(int x, int y)
{
	printf(" %c --> %c\n", x, y);
}
void Hanoi(int n, char a, char b, char c)
{
	if (n == 1)
	{
		Print_Move(a, c);
	}
	else
	{
	    //第一步,从a借助c移到b
		Hanoi(n - 1, a, c, b);//形参的前后两个分别是发出地和目的地
		//第二步:把最后的大盘子从a移到c
		Print_Move(a, c);
		//第三步:从b借助a移到c
		Hanoi(n - 1, b, a, c);
	}
}
int main()
{
	int n = 0;
	char A;
	char B;
	char C;
	while (scanf("%d", &n) != EOF)
	{
		Hanoi(n, 'A', 'B','C');//传参:盘子数,三根柱子代号
	}
	return 0;
}

函数的内核逻辑还是铁三步:
1.从a借助c移到b
2.把最后的大盘子从a移到c
3.从b借助a移到c
在这里插入图片描述

在这里插入图片描述
递归的轨迹,这张图十分清晰

如果要数移过的次数:

#include<stdio.h>
int count=0;
void Hanoi(int n, char a, char b, char c)
{
	if (n >= 2)
	{
		Hanoi(n - 1, a, c, b);
		count++;
		printf("%d: %c -> %c\n",count,a,c);
		Hanoi(n - 1, b, a, c);
	}
	else if (n == 1)
	{
		count++;
		printf("%d: %c -> %c\n", count, a, c);
	}
}
int main()
{
	int n = 0;
	scanf("%d", &n);
	Hanoi(n, 'A', 'B', 'C');
	return 0;
}

这里,我把图中的Move函数命名为了Print_Move:
因为这个函数的作用仅仅是死板地打印“ A–>C" 来让整个过程呈现的完整一些
完全可以不在两个Hanoi()函数之间加这个Print_Move
因为,这里的逻辑主要是前面n-1个盘子的位移,就算不动最后一个盘子也毫无影响,它是最大的,任何一个盘子都可以摞在它上面

4.来点料!汉诺塔的背景

相传在世界中心贝拿勒斯(在印度北部)的圣庙里,一块黄铜板上插着三根宝石针。印度教的主神梵天在创造世界的时候,在其中一根针上从下到上地穿好了由大到小的64片金片 当所有的金片都从梵天穿好的那根针上移到另外一根针上时,世界就将在一声霹雳中消灭,而梵塔、庙宇和众生也都将同归于尽。
———————————————————————————————
那么古代印度距离现在已经很远,这64个圆盘还没移动完么?
我们来通过计算来看看要完成这个任务到底要多少时间

上面笔者已经推出了通项公式
当n为64时
2^64-1=18446744073709551615(次)
假设 1 秒移 1 个
一年为31536000秒
18446744073709551615/31536000约是 584942417355 天
约是 5845.54 亿年
太阳的完整寿命大约100亿年
目前约过了50亿年
所以整个人类文明都等不到那天

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值