学习笔记:栈的应用1_递归(重点)

栈的应用1_递归(重点)

1. 函数调用栈

函数调用栈一般是从高地址向低地址增长的,栈底为内存的高地址处,栈顶为内存的低地址处

函数调用栈中存储的数据为活动记录,活动记录是函数调用时一系列相关信息的记录

程序的栈空间可以看做一个顺序栈的应用,栈保存了一个函数调用时所需要的维护信息,包括函数参数,函数返回地址,局部变量, 函数调用上下文。

在不断的压栈过程中造成栈空间耗尽而产生栈溢出,栈溢出常由于函数递归过深或局部数组过大而造成。

2. 递归的定义

​ 在高级语言中,调用自己和其它函数并没有本质的不同。我们把一个直接调用自己或通过一系列的调用语句间接地调用自己的函数,称做递归函数。

每个递归定义必须至少有一个条件,满足时递归不再进行,即不再引用自身而是返回值退出。

3. 递归的实例

(1)递归求n的阶乘
#include<stdio.h> 

long fact(int n)
{	
	long t;
 	if(n==0||n==1) t=1;
 	else t=n*fact(n-1);
 	return t;
}

int main()
{	
	int n=4;
	long f; 
	f= fact(n);
 	printf("%d!=%ld\n",n,f );
 	return 0;
}

分析:

在这里插入图片描述

(2)斐波那契数列

斐波那契数列:1,1,2,3,5,8,13… (前相邻两项之和,构成了后一项)

在这里插入图片描述

问题:打印出前40位的斐波那契数列

解法1:迭代法
int main()
{
    int i;
    int a[40];
    a[0] = 0;
    a[1] = 1;
    printf("%d ",a[0]);
    printf("%d ",a[1]);
    for(i=2;i<40;i++){
        a[i] = a[i-1]+a[i-2];
        printf("%d ",a[i]);
    }
    return 0;
}
解法2:递归法
int Fbi(int i)
{
	if(i<2)
		return i == 0 ? 01;
	return Fbi(i-1)+Fbi(i-2);
}

int main(){
	int i;
	printf("递归显示斐波那契数列:\n");
	for(i=0;i<40;i++){
		printf("%d ",Fbi(i))
	}
	return 0;
}

分析:模拟i=5的执行过程

在这里插入图片描述

对比两种实现斐波那契的代码。递归和迭代的区别是:迭代使用的是循环结构,递归使用的是选择结构。递归能使程序的结构更清晰,更间洁,更容易让人理解,从而减少读代码的时间。但是大量的递归调用会建立函数副本,会耗费大量的时间和内存。迭代则不需要反复调用函数和占用额外的内存。因此我们应该视不同情况选择不同的代码实现方式。

(3)n阶Hanoi塔问题

问题描述:假设有三个分别命名为A,B,C的塔座,在塔座x上插有n个直径大小各不相同,且从小到大编号分别为1,2,…,n的圆盘,现要求将塔座A上的n个圆盘借助塔柱B移动到塔柱C,且仍按相同顺序叠排,圆盘移动时需遵循以下规则:

1) 每次只能移动一个圆盘

2)圆盘可以插在A,B,C中的任何一个塔座上

3)任何时刻不能将较大的圆盘压在较小的圆盘上

在这里插入图片描述

#include<stdio.h>

void move(char A, char B){
	printf("%c -> %c\n",A,B);
}
void hanoi(int n,char A,char B,char C){
	if(n==1){
		move(A,C);
	}	
	else{
		hanoi(n-1,A,C,B);
		move(A,C);
		hanoi(n-1,B,A,C);
	}
}

int main(){
	hanoi(2,'A','B','C');
	
	return 0;
}

在这里插入图片描述
但行好事,莫问前程。
不忘初心,方得始终。
做好自己,温暖且积极!!

  • 5
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

程序猿小张的日常笔记

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值