广告
返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >c语言怎么循环加数组实现汉诺塔
  • 947
分享到

c语言怎么循环加数组实现汉诺塔

2023-06-29 01:06:47 947人浏览 泡泡鱼
摘要

今天小编给大家分享一下C语言怎么循环加数组实现汉诺塔的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。简介汉诺塔问题是学数据结构

今天小编给大家分享一下C语言怎么循环加数组实现汉诺塔的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。

简介

汉诺塔问题是学数据结构与算法的时候会遇到的问题,相信来看本文的读者应该都对汉诺塔问题有基本的了解,理论上所有的递归都可以改成循环,常规的做法是借助堆栈,但是我一想好像用循环加数组也可以实现,于是就有了本文,实现声明,本文最后出来的算法效率不高的,比直接用递归实现还要差很多,追求算法效率的同学就不用看这个了。
题目:假设有3个柱子,分别为A、B、C,A柱子上有数量为n个的空心圆盘,从上到下序号分别为1...n,要求把A柱子中的n个圆盘移动到C柱中,且序号大的盘子必须在在需要小的圆盘下方。
思路:如果只有1个圆盘需要移动,则直接从A柱移动到C柱,如果有n个圆盘(n>1)需要移动,则需要先把n-1个圆盘从A柱移动到B柱,再把第n个圆盘从A柱移动到C柱,最后把原来的n-1个圆盘从B柱移动到C柱。

例如:

将1个圆盘从A柱移动到C柱只需要1个步骤:

把圆盘1从A移动到C

将2个圆盘从A柱移动到C柱需要3个步骤:

把圆盘1从A移动到B
2. 把圆盘2从A移动到C
3. 把圆盘1从B移动到C

将3个圆盘从A柱移动到C柱需要7个步骤:

把圆盘1从A移动到C
2. 把圆盘2从A移动到B
3. 把圆盘1从C移动到B
4. 把圆盘3从A移动到C
5. 把圆盘1从B移动到A
6. 把圆盘2从B移动到C
7. 把圆盘1从A移动到C

递归的汉诺塔解法(c语言)

可以看出下面的递归算法的时间复杂度为O(2^n),因为存在有调用2^n-2次递归调用和1次原生printf;而空间复杂度为O(n),因为运行时栈中最多会同时保存n个函数的调用信息。

#include <stdio.h>#include <stdlib.h>#include <math.h>void towers(int n, char from, char to, char aux);int main() {    towers(3, 'A', 'C', 'B');    return 0;}void showMove (int order, char from, char to) {    printf("move disk %d from %c to %c\n", order, from, to);}void towers(int n, char from, char to, char aux) {    if (n==1) {        showMove(n, from, to);        return;    }    towers(n-1, from, aux, to);        showMove(n, from, to);    towers(n-1, aux, to, from);}

解释一下代码中参数的含义,void towers(int n, char from, char to, char aux)的意思是把n个圆盘从from柱子移动到to柱子,中间可以借用aux柱子。

分析一下上面的递归执行过程:

我们已经知道把1个圆盘从from移动到to的步骤是showMove(1, from, to);

而把2个圆盘从from移动到to的步骤是,先照着移动1个圆盘的操作,把前面1个圆盘从from移动到aux,再把第2个圆盘从from移动到to,最后按照移动1个圆盘的操作,把刚才的1个圆盘从aux移动到to。

同理,把3个圆盘从from移动到to的步骤是,先照着移动2个圆盘的操作,把前面2个圆盘从from移动到aux,再把第3个圆盘从from移动到to,最后按照移动2个圆盘的操作,把刚才的2个圆盘从aux移动到to。

所以,把n个圆盘的操作从from移动到to的步骤是,先照着移动n-1个圆盘的操作,把前面n-1个圆盘从from移动到aux,再把第n个圆盘从from移动到to,最后按照移动n-1个圆盘的操作,把刚才的n-1个圆盘从aux移动到to。

因此,我们可以记录移动1个圆盘的步骤,来得到移动2个圆盘的步骤,通过移动2个圆盘的步骤来得到移动3个圆盘的步骤,...最后得到移动n个圆盘的步骤。经过分析可以知道移动n个圆盘一共会有2^n-1个步骤

循环实现汉诺塔问题(c语言)

为了代码更加易懂,这里写了注释,修改了部分变量命名,统一用数组保存步骤集合,最后才输出。
可以看出这个算法的时间复杂度还是O(2^n),一共会执行2^(n+1)-1次setMoveAction语句和2^n-1次,printf语句,比起直接用递归还要复杂一些,而空间复杂度也是O(2^n),属于没什么用的算法,就是随便写写。

#include <stdio.h>#include <stdlib.h>#include <math.h>void towers(int n, char from, char to, char aux);int main(){    towers(3, 'A', 'C', 'B');    return 0;}void showMove(int order, char from, char to){    printf("move disk %d from %c to %c\n", order, from, to);}typedef struct{    int order;    char from;    char to;} MoveAction;void setMoveAction(MoveAction *p, int order, char from, char to){    p->order = order;    p->from = from;    p->to = to;}char compare_map(char c, char left, char right) {    if (c == left) {        return right;    } else if (c == right) {        return left;    }    return c;}void towers(int n, char from, char to, char aux){    MoveAction *actions, action;    int i, k, size;    char f, t;    actions = (MoveAction *)calloc(pow(2, n - 1) - 1, sizeof(MoveAction));    setMoveAction(&actions[0], 1, from, to);    size = 1;    for (i = 2; i <= n; i++)    {        //本次循环会将把i个盘子从from移动到to的步骤记录到actions数组中        // 设size是把i-1个盘子从from移动到to的步骤数,        // 则本次循环中集合{actions[x] | 0 <= x <= size -1 }就是size是把i-1个盘子从from移动到aux的步骤集合,        //而action[size]是把第i个盘子从from移到to,        //而集合{actions[x] | size + 1 <= x <= 2*size }就应该是把i-1个盘存从aux移动到to的步骤集合        // 倒序,先求解集合{actions[x] | size + 1 <= x <= 2*size }        for (k = 0; k < size; k++)        {            action = actions[k];            f = compare_map(action.from, from, aux);            t = compare_map(action.to, from, aux);            setMoveAction(&actions[k + size + 1], action.order, f, t);        }        // 求解action[size]        setMoveAction(&actions[size], i, from, to);        // 求解集合{actions[x] | 0 <= x <= size -1 }        for (k = 0; k < size; k++)        {            action = actions[k];            f = compare_map(action.from, to, aux);            t = compare_map(action.to, to, aux);            setMoveAction(&actions[k], action.order, f, t);        }        size = size * 2 + 1;    }    for (i = 0; i < size; i++)    {        showMove(actions[i].order, actions[i].from, actions[i].to);    }}

以上就是“c语言怎么循环加数组实现汉诺塔”这篇文章的所有内容,感谢各位的阅读!相信大家阅读完这篇文章都有很大的收获,小编每天都会为大家更新不同的知识,如果还想学习更多的知识,请关注编程网其他教程频道。

--结束END--

本文标题: c语言怎么循环加数组实现汉诺塔

本文链接: https://www.lsjlt.com/news/321677.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • c语言怎么循环加数组实现汉诺塔
    今天小编给大家分享一下c语言怎么循环加数组实现汉诺塔的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。简介汉诺塔问题是学数据结构...
    99+
    2023-06-29
  • c语言循环加数组实现汉诺塔问题
    目录简介递归的汉诺塔解法(c语言)循环实现汉诺塔问题(c语言)简介 汉诺塔问题是学数据结构与算法的时候会遇到的问题,相信来看本文的读者应该都对汉诺塔问题有基本的了解,理论上所有的递归...
    99+
    2022-11-13
  • C语言怎么实现汉诺塔
    这篇文章主要介绍了C语言怎么实现汉诺塔的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇C语言怎么实现汉诺塔文章都会有所收获,下面我们一起来看看吧。1.递归思想简介在c语言中,程序调用自身的编程技巧称为递归( re...
    99+
    2023-06-28
  • C语言用递归函数实现汉诺塔
    目录汉诺塔(Hanoi)是什么?那么,C语言如何实现汉诺塔呢?汉诺塔的基本思路是:具体代码见下(注意点在代码下面):总结汉诺塔(Hanoi)是什么? 一个简单的汉诺塔就如上图所示...
    99+
    2022-11-13
  • C语言怎么运用函数的递归实现汉诺塔
    这篇文章主要讲解了“C语言怎么运用函数的递归实现汉诺塔”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“C语言怎么运用函数的递归实现汉诺塔”吧!1、汉诺塔是如何实现的下面是有三个盘子的示例:从左...
    99+
    2023-07-02
  • C语言运用函数的递归实现汉诺塔
    目录1、汉诺塔是如何实现的2、汉诺塔问题画图详解3、汉诺塔问题代码解释总结1、汉诺塔是如何实现的 下面是有三个盘子的示例: 从左到右一次是 A柱 B柱 C柱 A柱:起始位置 B柱:...
    99+
    2022-11-13
  • C语言递归函数与汉诺塔问题怎么解决
    今天小编给大家分享一下C语言递归函数与汉诺塔问题怎么解决的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。递归函数直接或者间接调...
    99+
    2023-07-02
  • c语言实现数组循环左移m位
    目录c语言数组循环左移m位数组循环左移的简单方法输入格式输出格式c语言数组循环左移m位 #include<stdio.h> //函数原型 void rightshiftl...
    99+
    2022-11-13
  • c语言怎么用for循环给数组赋值
    在C语言中,可以使用for循环给数组赋值。以下是一个示例:```c#include #define SIZE 5int main()...
    99+
    2023-09-04
    c语言
  • C语言怎么实现循环双链表
    这篇文章主要介绍“C语言怎么实现循环双链表”,在日常操作中,相信很多人在C语言怎么实现循环双链表问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”C语言怎么实现循环双链表”的疑惑有所帮助!接下来,请跟着小编一起来...
    99+
    2023-06-25
  • C语言算法积累加tag的循环队列怎么实现
    这篇文章主要讲解了“C语言算法积累加tag的循环队列怎么实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“C语言算法积累加tag的循环队列怎么实现”吧!题目:若希望循环队列中的元素都能得到利...
    99+
    2023-06-30
  • C语言怎么实现带头双向循环链表
    本篇内容主要讲解“C语言怎么实现带头双向循环链表”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“C语言怎么实现带头双向循环链表”吧!创建链表存储结构我们需要创建一个结构体来存储一个链表结点的相关信...
    99+
    2023-06-30
  • C语言带头双向循环链表怎么实现
    这篇“C语言带头双向循环链表怎么实现”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“C语言带头双向循环链表怎么实现”文章吧。带...
    99+
    2023-06-30
  • C语言单双线性及循环链表怎么实现
    今天小编给大家分享一下C语言单双线性及循环链表怎么实现的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。链表思维顺序存储结构Op...
    99+
    2023-07-05
  • C语言链式队列与循环队列怎么实现
    这篇文章主要介绍了C语言链式队列与循环队列怎么实现的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇C语言链式队列与循环队列怎么实现文章都会有所收获,下面我们一起来看看吧。队列的实现队列是一种先进先出(First ...
    99+
    2023-06-30
  • C语言数据结构中双向带头循环链表怎么实现
    这篇文章主要讲解了“C语言数据结构中双向带头循环链表怎么实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“C语言数据结构中双向带头循环链表怎么实现”吧!一、概念来画张图总体回顾下:在我们学习...
    99+
    2023-06-30
  • C#怎么实现泛型动态循环数组队列
    这篇文章主要介绍“C#怎么实现泛型动态循环数组队列”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“C#怎么实现泛型动态循环数组队列”文章能帮助大家解决问题。任务循环数组实现目标:(1)创建一个新的数组...
    99+
    2023-06-29
  • c语言怎么实现数组的逆置
    可以利用两个指针来实现数组的逆置。一个指向数组的起始位置,一个指向数组的末尾位置,然后交换两个指针指向的元素,然后分别向数组中心移动...
    99+
    2023-10-28
    c语言
  • C语言怎么实现循环打印星号图形再镂空
    这篇“C语言怎么实现循环打印星号图形再镂空”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“C语言怎么实现循环打印星号图形再镂空...
    99+
    2023-07-04
  • C语言怎么实现两个整数相加
    这篇文章主要介绍“C语言怎么实现两个整数相加”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“C语言怎么实现两个整数相加”文章能帮助大家解决问题。使用 scanf() 来接收输入, printf() 与...
    99+
    2023-06-17
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作