详解递归(基础篇)———函数栈、阶乘、Fibonacci数列
一、递归的基本概念
递归函数:在定义的时候,自己调用了自己的函数。
注意:递归函数定义的时候一定要明确结束这个函数的条件!
二、函数栈
栈:一种数据结构,它仅允许栈顶进,栈顶出,先进后出,后进先出。我们可以简单的理解为栈就是一个杯子,这个杯子里面有很多隔层,每一层都可以放东西,第一个放入的东西就在杯子最后一层,第二个放入的东西就在倒数第二层,现在我们要取出最后一层的东西,就必须先把第二层的东西给出来。
函数栈:栈里面每一层都是装的都是函数的栈就是函数栈,调用一个函数的时候,这个函数就入栈,这个函数调用完成了(执行到了函数的最后一个语句或者说return了),就出栈。

下面是一个演示函数栈运行机制的C语言程序,并无实际意义,仅用于理解函数栈:
#include<stdio.h>
void function1()
{
printf("function1 done!")
return; //第三步,函数function1调用完毕,出栈
}
void function2()
{
printf("function2 done!")
return; //第五步,函数function2调用完毕,出栈
}
void function3()
{
printf("function3 done!")
return; //第八步,函数function3调用完毕,出栈
}
void function4()
{
printf("function4 done!")
function3(); //第七步,调用函数function3,入栈
return; //第九步,函数function4调用完成,出栈
}
int main()//第一步,调用主函数,主函数入栈(这个C语言程序的入口)
{
function1(); //第二步,调用函数function1,入栈
function2(); //第四步,调用函数function2,入栈
function4(); //第六步,调用函数function4,入栈
return 0; //第十步,主函数调用完成,出栈(整个程序执行完成)
}
三、实例
1、阶乘
题目:用递归方法实现计算整数n的阶乘n!
解析:
首先,我们易知0!=1; 1!=1; 2!=2*1!; 3!=3*2!; 4!=4*3!;……
那么我们可以得到递推公式
- n!=1[n=0,1]
- n!=n*(n-1)![n>=2]
从而我们就可以写出计算阶乘的递归程序的C语言代码,如下:
#include<stdio.h>
long Fact(int n);
int main()
{
int n;
long result;
printf("Input n:");
scanf("%d",&n);
result = Fact(n);
if(result == -1)
printf("n<0,data error!\n");
else
printf("%d! = %ld\n", n, result);
return 0;
}
long Fact(int n)
{
//对传入函数的值判断其合法性
if(n < 0)
{
return -1;
}
//计算n的阶乘的表达式改写的代码
else if(n == 0 || n == 1)
{
return 1;
}
else
{
return (n*Fact(n-1));
}
}
当然,写出来了并不代表我们理解,下图讲解了在这个程序中是如何计算4的阶乘的。

2、Fibonacci数列
题目:实现函数表达式Fib(0)=0,Fib(1)=1,Fib(n)=Fib(n-1)+Fib(n-2)。
解析:在这里题目已经给出了递归表达式,那么我们可以轻松地写出C语言代码。
#include<stdio.h>
int Fib(int n)
{
//判断传入数据的合法性
if(n < 0)
{
return -1;
}
//递归表达式改而写的C语言代码
else if(n == 0)
{
return 0;
}
else if(n == 1)
{
return 1;
}
else
{
return Fib(n-1)+Fib(n-2);
}
}
int main()
{
int n,result;
printf("Input n:");
scanf("%d",&n);
result = Fib(n);
if(result == -1)
{
printf("A illegal data!");
}
else
{
printf("Fib(%d) = %d", n, result);
}
return 0;
}
关于用Fibonacci数列的在上述程序中如何计算的,我们仍然是通过函数栈进行,具体如何运行在此就不再赘述。
详解递归(基础篇)———函数栈、阶乘、Fibonacci数列的更多相关文章
- SpringBoot系列教程JPA之query使用姿势详解之基础篇
前面的几篇文章分别介绍了CURD中的增删改,接下来进入最最常见的查询篇,看一下使用jpa进行db的记录查询时,可以怎么玩 本篇将介绍一些基础的查询使用姿势,主要包括根据字段查询,and/or/in/l ...
- hadoop之mapreduce详解(基础篇)
本篇文章主要从mapreduce运行作业的过程,shuffle,以及mapreduce作业失败的容错几个方面进行详解. 一.mapreduce作业运行过程 1.1.mapreduce介绍 MapRed ...
- 最简单MySQL教程详解(基础篇)之多表联合查询
在关系型数据库中,我们通常为了减少数据的冗余量将对数据表进行规范,将数据分割到不同的表中.当我们需要将这些数据重新合成一条时,就需要用到我们介绍来将要说到的表连接. 常用术语冗余(Redundancy ...
- hadoop之mapreduce详解(进阶篇)
上篇文章hadoop之mapreduce详解(基础篇)我们了解了mapreduce的执行过程和shuffle过程,本篇文章主要从mapreduce的组件和输入输出方面进行阐述. 一.mapreduce ...
- TCP协议详解(理论篇)
TCP协议详解(理论篇) 2012-08-20 0个评论 作者:陈立龙 收藏 我要投稿 TCP协议详解(理论篇) 1. 与UDP不同的是,TCP提供了一种面向连接 ...
- 第7.25节 Python案例详解:使用property函数定义与实例变量同名的属性会怎样?
第7.25节 Python案例详解:使用property函数定义与实例变量同名的属性会怎样? 一. 案例说明 我们上节提到了,使用property函数定义的属性不要与类内已经定义的普通实例变量重 ...
- IIS负载均衡-Application Request Route详解第四篇:使用ARR实现三层部署架构(转载)
IIS负载均衡-Application Request Route详解第四篇:使用ARR实现三层部署架构 系列文章链接: IIS负载均衡-Application Request Route详解第一篇: ...
- [转]ANDROID L——Material Design详解(动画篇)
转载请注明本文出自大苞米的博客(http://blog.csdn.net/a396901990),谢谢支持! 转自:http://blog.csdn.net/a396901990/article/de ...
- appledoc导出iOS代码文档的使用和问题详解(干货篇)
appledoc导出iOS代码文档的使用和问题详解(干货篇) 1. 简单说一下背景和自己感受 背景: 项目好像突然黄了,公司让详细写项目代码的注释并且导出文档,弄完之后就要封版. 说实话:听到这个消息 ...
- Android为TV端助力 转载:Android绘图Canvas十八般武器之Shader详解及实战篇(上)
前言 Android中绘图离不开的就是Canvas了,Canvas是一个庞大的知识体系,有Java层的,也有jni层深入到Framework.Canvas有许多的知识内容,构建了一个武器库一般,所谓十 ...
随机推荐
- JVM学习心得—JVM内存模型(个人整理,请勿转载)
一.运行时数据区域 线程私有的:程序计数器+虚拟机栈+本地方法栈 线程共享的:堆+方法区(运行时常量池)+直接内存(非运行时数据区的一部分) *JDK1.8后将方法区废除,新增元空间. 1.1 程序计 ...
- 分布式深度学习DDL解析
分布式深度学习DDL解析 一.概述 给一个庞大的GPU集群,在实际的应用中,现有的大数据调度器会导致长队列延迟和低的性能,该文章提出了Tiresias,即一个GPU集群的调度器,专门适应分布式深度学习 ...
- 如何在TVM上集成Codegen(上)
如何在TVM上集成Codegen(上) 许多常用的深度学习内核,或者提供DNNL或TensorRT等框架和图形引擎,让用户以某种方式描述他们的模型,从而获得高性能.此外,新兴的深度学习加速器也有自己的 ...
- 特斯拉Tesla Model 3整体架构解析(上)
特斯拉Tesla Model 3整体架构解析(上) 一辆特斯拉 Model 3型车在硬件改造后解体 Sensors for ADAS applications 特斯拉 Model 3型设计的传感器组件 ...
- spring——自动装配【非常详细】
什么是自动装配? 自动装配就是让应用程序上下文为你找出依赖项的过程.说的通俗一点,就是Spring会在上下文中自动查找,并自动给bean装配与其关联的属性! spring中实现自动装配的方式有两种,一 ...
- springboot——发送put、delete请求
在springmvc中我们要发送put和delete请求,需要先配置一个过滤器HiddenHttpMethodFilter,而springboot中,已经帮我们自动配置了,所以我们可以不用配置这个过滤 ...
- 【NX二次开发】三点画圆,三角形外心,已知三点求圆心
已知P1.P2.P3,求点O 算法:三点不在一条直线上时,通过连接任意两点,作中垂线.任意两条中垂线的交点是圆心.
- SpringCloud(1)生态与简绍
一:微服务架构简绍学习目标 1.技术架构的演变,怎么一步步到微服务的:2.什么是微服务,优点与缺点 :3.SOA(面向服务)与MicroServices(微服务)的区别 :4.Dubbo 与Spri ...
- Redis源码解析之跳跃表(三)
我们再来学习如何从跳跃表中查询数据,跳跃表本质上是一个链表,但它允许我们像数组一样定位某个索引区间内的节点,并且与数组不同的是,跳跃表允许我们将头节点L0层的前驱节点(即跳跃表分值最小的节点)zsl- ...
- MySQL数据的高效检索
数据库操作中,常常需要完成既定数据的检索.少量数据存放在表中,只需使用基本的SQL语句即可检索得到.但当数据量较大时,受MySQL数据库底层实现原理的限制,缺省的SQL语句,检索效率较低. 例如:当执 ...