函数中有直接或间接地调用自身函数的语句,这样的函数称为递归函数。递归函数用

得好,可简化编程工作。但函数自己调用自己,有可能造成死循环。为了避免死循环,要

做到两点:

(1) 降阶。递归函数虽然调用自身,但并不是简单地重复。它的实参值每次是不一样

的。一般逐渐减小,称为降阶。如教科书式(33)的Ackerman 函数,当m≠0 时,求

Ack(m,n)可由Ack(m-1,⋯)得到,Ack()函数的第1 个参数减小了。

(2) 有出口。即在某种条件下,不再进行递归调用。仍以教科书式(33)的Ackerman

函数为例,当m=0 时,Ack(0,n)=n+1,终止了递归调用。所以,递归函数总有条件语

句。m=0 的条件是由逐渐降阶形成的。如取Ack(m,n)函数的实参m=-1,即使通过降阶也

不会出现m=0 的情况,这就造成了死循环。

编译软件在遇到函数调用时,会将调用函数的实参、返回地址等信息存入软件自身所

带的栈中,再去运行被调函数。从被调函数返回时,再将调用函数的信息出栈,接着运行

调用函数。编译软件开辟的栈空间是有限的,当递归调用时,嵌套的层次往往很多,就有

可能使栈发生溢出现象,从而出现不可预料的后果。运行algo3-8.cpp 时,m、n 的取值就

不可过大。

// algo3-8.cpp 用递归调用求Ackerman(m,n)的值
#include<stdio.h>
int ack(int m,int n)
{
int z;
if(m==0)
z=n+1; // 出口
else if(n==0)
z=ack(m-1,1); // 对形参m降阶
else
z=ack(m-1,ack(m,n-1)); // 对形参m、n降阶
return z;
}
void main()
{
int m,n;
printf("Please input m,n:");
scanf("%d,%d",&m,&n);
printf("Ack(%d,%d)=%d\n",m,n,ack(m,n));
}

代码运行结果:

/*
Please input m,n:3,9
Ack(3,9)=4093
Press any key to continue */
// algo3-9.cpp 用递归函数求解迷宫问题(求出所有解)
#include"c1.h" // 根据《PASCAL程序设计》(郑启华编著)中的程序改编
#include"func3-1.cpp" // 定义墙元素值为0,可通过路径为-1,通过路径为足迹
void Try(PosType cur,int curstep)
{ // 由当前位置cur、当前步骤curstep试探下一点
int i;
PosType next; // 下一个位置
PosType direc[4]={{0,1},{1,0},{0,-1},{-1,0}}; // {行增量,列增量},移动方向,依次为东南西北
for(i=0;i<=3;i++) // 依次试探东南西北四个方向
{
next.x=cur.x+direc[i].x; // 根据移动方向,给下一位置赋值
next.y=cur.y+direc[i].y;
if(m[next.x][next.y]==-1) // 下一个位置是通路
{
m[next.x][next.y]=++curstep; // 将下一个位置设为足迹
if(next.x!=end.x||next.y!=end.y) // 没到终点
Try(next,curstep); // 由下一个位置继续试探(降阶递归调用,离终点更近)
else // 到终点
{
Print(); // 输出结果(出口,不再递归调用)
printf("\n");
}
m[next.x][next.y]=-1; // 恢复为通路,以便在另一个方向试探另一条路
curstep--; // 足迹也减1
}
}
}
void main()
{
Init(-1); // 初始化迷宫,通道值为-1
printf("此迷宫从入口到出口的路径如下:\n");
m[begin.x][begin.y]=1; // 入口的足迹为1
Try(begin,1); // 由第1步入口试探起
}

代码运行结果如下:

此迷宫从入口到出口有2 条路径。由入口(1 行1 列足迹1)处向4 个方向试探,只有2

个方向(东、南)是通路(-1)。首先朝东继续试探。足迹2~4 都只有1 个方向是通路,足

迹5 到达出口。输出路径且逐一恢复足迹为-1(通路)。恢复后的情况是除入口为1 外,迷

宫其它各点与初态相同,以便第2 条路径(由入口向南)继续试探。

// algo3-10.cpp Hanoi塔问题,调用算法3.5的程序
#include<stdio.h>
int c=0; // 全局变量,搬动次数
void move(char x,int n,char z)
{ // 第n个圆盘从塔座x搬到塔座z
printf("第%i步: 将%i号盘从%c移到%c\n",++c,n,x,z);
}
void hanoi(int n,char x,char y,char z) // 算法3.5
{ // 将塔座x上按直径由小到大且自上而下编号为1至n的n个圆盘
// 按规则搬到塔座z上。y可用作辅助塔座
if(n==1) // (出口)
move(x,1,z); // 将编号为1的圆盘从x移到z
else
{
hanoi(n-1,x,z,y); // 将x上编号为1至n-1的圆盘移到y,z作辅助塔(降阶递归调用)
move(x,n,z); // 将编号为n的圆盘从x移到z
hanoi(n-1,y,x,z); // 将y上编号为1至n-1的圆盘移到z,x作辅助塔(降阶递归调用)
}
}
void main()
{
int n;
printf("3个塔座为a、b、c,圆盘最初在a座,借助b座移到c座。请输入圆盘数:");
scanf("%d",&n);
hanoi(n,′a′,′b′,′c′);
}

代码运行结果如下:

/*
3个塔座为a、b、c,圆盘最初在a座,借助b座移到c座。请输入圆盘数:3
第1步: 将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
Press any key to continue */

栈与递归的实现(Hanoi塔问题等等)的更多相关文章

  1. Hanoi塔问题——递归

    /////////////Hanoi塔问题///////#include<iostream>using namespace std;void hanoi(int i,char A,char ...

  2. 栈->栈与递归

    文字简述 1.阶乘函数 2.2阶Fiibonacci数列 3.n阶Hanoi塔问题 代码实现 // // Created by lady on 19-4-3. // #include <stdi ...

  3. c++迭代递归实现汉诺塔(5种迭代方法满足你)

    #include <iostream> //从A到C using namespace std; int n; void ready() { cout << "请输入汉 ...

  4. 经典递归算法研究:hanoi塔的理解与实现

    关于hanoi塔的原理以及概念,请Google,访问不了去百度. 主要设计到C中程序设计中递归的实现: 主代码实现如下: void hanoi(int src, int dest, int tmp, ...

  5. (转)Hanoi塔问题分析

    转自:http://shmilyaw-hotmail-com.iteye.com/blog/2077098 简介 关于Hanoi塔问题的分析,在网上的文章都写烂了.之所以打算写这篇文章,更多的是针对这 ...

  6. Python递归实现汉诺塔

    Python递归实现汉诺塔: def f3(n,x,y,z): if(n==1): print(x,'--->',z) else: f3(n-1,x,z,y) print(x,'--->' ...

  7. 剑指offer_面试题5_从尾到头打印链表(栈和递归实现)

    题目:输入一个链表的头结点,从尾到头反过来打印出每一个节点的值 考察 单链表操作.栈.递归等概念. 理解:要实现单链表的输出,那么就须要遍历.遍历的顺序是从头到尾.而节点输出的顺序是从尾到头.因此,先 ...

  8. 深度优先搜索入门:POJ1164城堡问题(递归、用栈模拟递归)

    将问题的各状态之间的转移关系描述为一个图,则深度优先搜索遍历整个图的框架为:Dfs(v) {if( v 访问过)return;将v标记为访问过;对和v相邻的每个点u: Dfs(u);}int main ...

  9. UVA10410 TreeReconstruction 树重建 (dfs,bfs序的一些性质,以及用栈处理递归 )

    题意,给你一颗树的bfs序和dfs序,结点编号小的优先历遍,问你可能的一种树形: 输出每个结点的子结点. 注意到以下事实: (1)dfs序中一个结点的子树结点一定是连续的. (2)bfs,dfs序中的 ...

随机推荐

  1. 接口和lambda表达式笔记

    接口 接口是双方,即服务提供方和想让它们的对象对服务是可用的那些类,之间约定的一种机制. 声明一个接口 public interface IntSequence{ //不提供实现,则该方法为抽象方法, ...

  2. Druid单机环境安装指南

    1.下载单机环境必备工具 下载druid-0.10.1-bin.tar.gz和tranquility-distribution-0.8.2.tgz插件 http://druid.io/download ...

  3. WPF中的命令与命令绑定(二)

    原文:WPF中的命令与命令绑定(二) WPF中的命令与命令绑定(二)                                              周银辉在WPF中,命令(Commandi ...

  4. Django学习之天气调查实例(2):显示数据表数据

    数据表数据添加后,如添加3条用户信息,分别为“aaa”.“bbb”.“ccc”,现在通过代码的方式显示数据表中的数据. 1.在website项目文件夹中创建 userload.py文件,并且写如下代码 ...

  5. Android开发——View绘制过程源码解析(一)

    )UNSPECIFIED:表示View可以设置成任意的大小,没有任何限制.这种情况比较少见. 2. MeasureSpec的生成过程 2.1 顶级View的MeasureSpec // desired ...

  6. vs13发布web程序 iis上

    一.配置iis 1,找到控制面板--程序--启用或关闭Windows功能 2,从列表中选择Internet Infomation Services,并且把相应的功能条目勾选上,如果不清楚,可以全部选中 ...

  7. linux命令大全(转载)

    在搭建openstack时遇到问题,导致上网查询相关信息.找到一篇不错的文章,希望对大家有用.下附地址: http://blog.csdn.net/junbujianwpl/article/detai ...

  8. 今日Linux下安装部署禅道

    我的linux系统是在虚拟机上安装的Ubuntu,禅道在官网www.zentao.net下载安装的开源版的linux64位,采用一键安装包安装.安装前要求:系统上不能有自己安装的mysql .下载的安 ...

  9. json格式化显示样式js代码分享

    最近开发中需要在页面展示json.特整理了下代码,送给大家,希望能帮到有同样需求的朋友们. 代码: <html> <script src="http://cdn.bootc ...

  10. LeetCode 3——无重复字符的最长子串

    1. 题目 2. 解答 2.1. 方法一 我们从前往后遍历字符串,start 代表最长子串的起始位置,一开始设置为零. 如果没有遇到重复字符,则更新子串的长度,向后遍历. 如果遇到重复字符时,则更新字 ...