递归函数

递归是什么

递归其实是⼀种解决问题的⽅法,在C语⾔中,递归就是函数⾃⼰调⽤⾃⼰。写⼀个史上最简单的C语⾔递归代码:

#include <stdio.h>

int main() {
	printf("hehe\n");
	main();
	return 0;
}

上述就是⼀个简单的递归程序,只不过上⾯的递归只是为了演⽰递归的基本形式,不是为了解决问 题,代码最终也会陷⼊死递归,导致栈溢出(Stack  overflow)。

递归的思想:

把⼀个⼤型复杂问题层层转化为⼀个与原问题相似,但规模较⼩的⼦问题来求解;直到⼦问题不能再被拆分,递归就结束了。所以递归的思考⽅式就是把⼤事化⼩的过程。递归中的递就是递推的意思,归就是回归的意思。

递归的限制条件

递归在书写时,有两个必要条件

递归存在限制条件,当满足这个限制条件的时候,递归不在继续。

每次递归调用之后越来越接近这个限制条件。

递归举例

求n的阶乘

int Fact(int n) {
	if (n == 0) {
		return 1;
	}
	else {
		return n * Fact(n - 1);
	}
}

int main() {
	int sun = 0;
	int a = 5;
	sun = Fact(a);
	printf("%d",sun);
}

这里不考虑n太大,太大会出现溢出。

图画演示

顺序打印一个整数的每一位

输入一个整数m,按照顺序打印整数的每一位

比如:输入1234 输出 1 2 3 4

自我分析一下,如何得到这个数的每一位,如果n是一位数,n的每一位就是自己,如果n超过一位,就要拆分每一位。1234%10就能得到4,然后1234/10得到123,就去掉了4,然后继续对123%10,得到了3,再除10去掉了3,以此类推就会得到每一位数字

void Print(int n) {
	if (n>9) {
		Print(n / 10);
	}
	printf("%d ", n % 10);
	
}

int main() {
	int m = 0;
	scanf("%d", &m);
	Print(m);
	return 0;
}

画图演示

递归与迭代

递归是⼀种很好的编程技巧,但是和很多技巧⼀样,也是可能被误⽤的,就像举例1⼀样,看到推导的 公式,很容易就被写成递归的形式:

 int Fact(int n) 
{
    if(n==0)
        return 1;
    else
        return n*Fact(n-1);
}

Fact函数是可以产⽣正确的结果,但是在递归函数调⽤的过程中涉及⼀些运⾏时的开销。在C语⾔中每⼀次函数调⽤,都需要为本次函数调⽤在内存的栈区,申请⼀块内存空间来保存函数调 ⽤期间的各种局部变量的值,这块空间被称为运⾏时堆栈,或者函数栈帧。函数不返回,函数对应的栈帧空间就⼀直占⽤,所以如果函数调⽤中存在递归调⽤的话,每⼀次递归 函数调⽤都会开辟属于⾃⼰的栈帧空间,直到函数递归不再继续,开始回归,才逐层释放栈帧空间。 所以如果采⽤函数递归的⽅式完成代码,递归层次太深,就会浪费太多的栈帧空间,也可能引起栈溢出(stackoverflow)的问题。

所以如果不想使⽤递归,就得想其他的办法,通常就是迭代的⽅式(通常就是循环的⽅式)。⽐如:计算n的阶乘,也是可以产⽣1~n的数字累计乘在⼀起的。

int Fact(int n) {
	int i = 0;
	int ret = 1;
	for (i = 1;i <= n;i++) {
		ret *= i;
	}
	return ret;
}

int main() {
	int m = 0;
	int n = 0;
	scanf("%d", &m);
	n = Fact(m);
	printf("%d", n);
}

上述代码是能够完成任务,并且效率是⽐递归的⽅式更好的。事实上,我们看到的许多问题是以递归的形式进⾏解释的,这只是因为它⽐⾮递归的形式更加清晰, 但是这些问题的迭代实现往往⽐递归实现效率更⾼。当⼀个问题⾮常复杂,难以使⽤迭代的⽅式实现时,此时递归实现的简洁性便可以补偿它所带来的运⾏时开销。

求第n个斐波那契数

斐波那契数列(Fibonacci sequence)‌,又称‌黄金分割数列,因数学家‌莱昂纳多·斐波那契(Leonardo Fibonacci)以兔子繁殖为例子而引入,故又称“兔子数列”。这个数列从第3项开始,每一项都等于前两项之和。

用递归方式实现

int FibDG(int n) {
	if (n <= 2) {
		return 1;
	}
	else {
		return Fib(n - 1) + Fib(n - 2);
	}
}

用迭代方式实现

int FibDD(int n) {
	int a = 1;
	int b = 1;
	int c = 1;
	if (n > 2) {
		c =  a + b;
		a = b;
		b = c;
		n--;
	}
	return c;
}

迭代的⽅式去实现这个代码,效率就要⾼出很多了。有时候,递归虽好,但是也会引⼊⼀些问题,所以我们⼀定不要迷恋递归,适可⽽⽌就好。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值