递归其实是⼀种解决问题的⽅法,在C语⾔中,递归就是
函数⾃⼰调⽤⾃⼰
。
1.递归的思想:
把⼀个⼤型复杂问题层层转化为⼀个与原问题相似,但规模较⼩的⼦问题来求解;直到⼦问题不再
被拆分,递归就结束了。所以递归的思考⽅式就是把⼤事化⼩的过程。
递归中的
递就是递推的意思,归就是回归
的意思。
2.递归的限制条件
递归在书写的时候,有2个必要条件:
•
递归存在限制条件,当满⾜这个限制条件的时候,递归便不再继续。
•
每次递归调⽤之后越来越接近这个限制条件。
在C语⾔中每⼀次函数调⽤,都要需要为本次函数调⽤在栈区申请⼀块内存空间来保存函数调⽤期间的各种局部变量的值,这块空间被称为运⾏时堆栈,或者函数栈帧。
函数不返回,函数对应的栈帧空间就⼀直占⽤,所以如果函数调⽤中存在递归调⽤的话,每⼀次递归函数调⽤都会开辟属于⾃⼰的栈帧空间,直到函数递归不再继续,开始回归,才逐层释放栈帧空间。所以如果采⽤函数递归的⽅式完成代码,递归层次太深,就会浪费太多的栈帧空间,也可能引起栈溢出(stack overflow)的问题。
3.求n!
![](https://img-blog.csdnimg.cn/direct/22cecaf317a341f5a328840d6fdbf7bc.png)
4.求n^k
![](https://img-blog.csdnimg.cn/direct/ff9c903e2e084fbf9b3427601d03ec6a.png)
5.求第n个斐波那契数
![](https://img-blog.csdnimg.cn/direct/fa990cbd619f4ff2ac4b14c9f35d7b0d.png)
![](https://img-blog.csdnimg.cn/direct/7ea0476b86344db5822f8a82b3582f64.png)
但是由例可以发现,函数递归在解决问题中并不是一直都是最适合的,要根据具体的问题来运用,在用函数递归求第n个斐波那契数的时候,当我们求的n过大时,有可能会导致栈溢出。且递归层次越深,重复计算的就越多,运行时的效率就越低,这时我们可以用迭代(循环)的方式。
事实上,我们看到的许多问题是以递归的形式进⾏解释的,这只是因为它⽐⾮递归的形式更加清晰,但是这些问题的迭代实现往往⽐递归实现效率更⾼。
当⼀个问题⾮常复杂,难以使⽤迭代的⽅式实现时,此时递归实现的简洁性便可以补偿它所带来的运⾏时开销。
6. 青蛙跳台阶问题
问题1:
(与求第n个斐波那契数问题相似)
![](https://img-blog.csdnimg.cn/direct/5457a79725104b989e512e420aaf7385.png)
![](https://img-blog.csdnimg.cn/direct/e7aa3680accc4709b3b594b773abbbde.png)
![](https://img-blog.csdnimg.cn/direct/87303f2af21f4ed7a2f90263d0d262fa.png)
问题2:
![](https://img-blog.csdnimg.cn/direct/70319ecff19c483596dafc597dc7d7b2.png)
分析:青蛙想要跳到第n个台阶上,可以从第n-1个台阶上跳一个台阶,或者从n-2个台阶上跳2个台阶,...或者从0个台阶上跳n个台阶。
A(n)=A(n-1)+A(n-2)+A(n-3)+...+A(1);
A(n-1)=A(n-2)+A(n-2)+A(n-3)+...+A(1);
A(n)=2A(n-1);
![](https://img-blog.csdnimg.cn/direct/26856df1a4a047b6afe60e2507a93ed6.png)
问题3:
分析:青蛙想要跳到第n个台阶上,可以从第n-1个台阶上跳一个台阶,或者从n-2个台阶上跳2个台阶,...或者从n-m个台阶上跳m个台阶。
A(n)=A(n-1)+A(n-2)+A(n-3)+...+A(n-m);
A(n-1)=A(n-2)+A(n-2)+A(n-3)+...+A(n-m-1);
A(n)=2A(n-1)-A(n-m+1);