【C语言】用函数递归的方法解决汉诺塔问题

简介: 【C语言】用函数递归的方法解决汉诺塔问题

1.问题起源:

汉诺塔(又称河内塔)问题是印度的一个古老的传说。开天辟地的神勃拉玛在一个庙里留下了三根金刚石的棒,第一根上面套着64个圆的金片,最大的一个在底下,其余一个比一个小,依次叠上去,庙里的众僧不倦地把它们一个个地从这根棒搬到另一根棒上,规定可利用中间的一根棒作为帮助,但每次只能搬一个,而且大的不能放在小的上面。面对庞大的数字(移动圆片的次数)18446744073709551615,看来,众僧们耗尽毕生精力也不可能完成金片的移动。后来,这个传说就演变为汉诺塔游戏。

2.汉诺塔游戏规则:

简单来说,就是把a柱上的n个圆盘,通过b柱作为辅助全部搬运到c柱上去。在搬运的过程中一次只能搬运一个圆盘,而且大圆盘不能放到小圆盘上面。

3.递归是什么?

要用递归的方法解决这个问题,我们首先要知道何为递归。

递归就是一个函数在它的函数体内调用它自身。执行递归函数将反复调用其自身,每调用一次就进入新的一层。递归函数必须有结束条件。

当函数在一直递推,直到遇到墙后返回,这个墙就是结束条件。

所以递归要有两个要素,结束条件与递推关系。

4.问题分析:

我们很容易知道:

当n=1时,我们直接将盘子从a柱移到c柱便解决了问题。

当n=2时,我们可以先把a柱上的第一个盘子移动到b柱(a柱最上面的盘子编号为1,向下依次递增),然后将a柱上的第二个盘子移动到c柱,再将b柱上的盘子移动到c柱即可。

但是当n的值为3及以上时问题便变得麻烦起来了。那么我们能否将后面复杂的问题化简为前面简单的两种情况呢?

答案是肯定的,这也是函数递归的目的:将复杂的问题简单化。

代码实现:

我们可以先通过动图先来看一下具体怎么实现。

汉诺塔问题(函数递归)

我们只需将盘子个数为n的分为两类解决即可:

当n=1时,将盘子从a->c即可。

当n>1时,将a柱上的n-1个盘子移动到b柱上,再将a柱上剩下的一个(也就是第n个)移动到c柱上,然后将b柱上的n-1个移动到,c柱上即可。(记住我们这里是将n-1个盘子看为一个整体)

#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
//汉诺塔问题
int count = 0;
void move(int n, char a, char b, char c)
{
  if (n == 1)
    printf("第%d次:%c->%c\n", ++count, a, c);
  else
  {
    move(n - 1, a, c, b);
    printf("第%d次:%c->%c\n", ++count, a, c);
    move(n - 1, b, a, c);
  }
}
int main()
{
  int n = 0;
  printf("请输入圆盘的个数:\n");
  scanf("%d", &n);
  move(n, 'a', 'b', 'c');
  printf("共移动了%d次\n", count);
  return 0;
}

测试结果如下:

5.结果分析:

我们会发现,当圆盘为1的时候只用移动一次,但当圆盘为4时,要15次,可想而知,这个问题设计计算量是很大的,而且还会发现,当圆盘个数特别大时,电脑运算速度会明显降低,因为计算机要通过不停进行递归,所以递归只是解决题目的一种方法。在我看来汉诺塔问题包括大多数能用函数递归解决的问题都需要我们有一种能力,那就是将一些东西当作一个整体来看待。比如汉诺塔问题中我们就要将n-1看作一个整体,如果不能做到有这种理解能力,那么将很难理解如何用函数递归解决汉诺塔问题。

相关文章
|
19天前
|
存储 C语言 开发者
【C语言】字符串操作函数详解
这些字符串操作函数在C语言中提供了强大的功能,帮助开发者有效地处理字符串数据。通过对每个函数的详细讲解、示例代码和表格说明,可以更好地理解如何使用这些函数进行各种字符串操作。如果在实际编程中遇到特定的字符串处理需求,可以参考这些函数和示例,灵活运用。
39 10
|
19天前
|
存储 程序员 C语言
【C语言】文件操作函数详解
C语言提供了一组标准库函数来处理文件操作,这些函数定义在 `<stdio.h>` 头文件中。文件操作包括文件的打开、读写、关闭以及文件属性的查询等。以下是常用文件操作函数的详细讲解,包括函数原型、参数说明、返回值说明、示例代码和表格汇总。
42 9
|
19天前
|
存储 Unix Serverless
【C语言】常用函数汇总表
本文总结了C语言中常用的函数,涵盖输入/输出、字符串操作、内存管理、数学运算、时间处理、文件操作及布尔类型等多个方面。每类函数均以表格形式列出其功能和使用示例,便于快速查阅和学习。通过综合示例代码,展示了这些函数的实际应用,帮助读者更好地理解和掌握C语言的基本功能和标准库函数的使用方法。感谢阅读,希望对你有所帮助!
31 8
|
19天前
|
C语言 开发者
【C语言】数学函数详解
在C语言中,数学函数是由标准库 `math.h` 提供的。使用这些函数时,需要包含 `#include <math.h>` 头文件。以下是一些常用的数学函数的详细讲解,包括函数原型、参数说明、返回值说明以及示例代码和表格汇总。
41 6
|
19天前
|
存储 C语言
【C语言】输入/输出函数详解
在C语言中,输入/输出操作是通过标准库函数来实现的。这些函数分为两类:标准输入输出函数和文件输入输出函数。
110 6
|
19天前
|
存储 缓存 算法
【C语言】内存管理函数详细讲解
在C语言编程中,内存管理是至关重要的。动态内存分配函数允许程序在运行时请求和释放内存,这对于处理不确定大小的数据结构至关重要。以下是C语言内存管理函数的详细讲解,包括每个函数的功能、标准格式、示例代码、代码解释及其输出。
49 6
|
19天前
|
C语言 开发者
【C语言】断言函数 -《深入解析C语言调试利器 !》
断言(assert)是一种调试工具,用于在程序运行时检查某些条件是否成立。如果条件不成立,断言会触发错误,并通常会终止程序的执行。断言有助于在开发和测试阶段捕捉逻辑错误。
27 5
|
4月前
|
机器学习/深度学习 C语言
【C语言篇】递归详细介绍(基础概念习题及汉诺塔等进阶问题)
要保持最小的步数,每一次汉诺塔问题(无论是最初还是递归过程中的),如果此时初始柱盘子数为偶数,我们第一步是把最上面的盘子移动到中转柱,如果为奇数,我们第一步则是将其移动到目标柱。
101 0
【C语言篇】递归详细介绍(基础概念习题及汉诺塔等进阶问题)
|
7月前
|
C语言
C语言递归问题【青蛙跳台阶】和【汉诺塔】
C语言递归问题【青蛙跳台阶】和【汉诺塔】
|
C语言
【C语言】递归详解汉诺塔问题
【C语言】递归详解汉诺塔问题
256 0
【C语言】递归详解汉诺塔问题