【算法】递归:递归优化之尾递归

引言:在以往我发过一篇过于通过分析法去理解递归求解递归的博客文章,那篇文章主要介绍了如何去求解递归问题。而在这篇文章中,我会介绍一下如何去优化递归,顺带还会去分析一下递归算法的性能,这篇文章的目的是一个小小的分享,希望大家能在此有收获。

摘要:文章首先会先通过斐波那契数列来复习在以往介绍的解决递归的方法,然后在通过这个例子去分析递归法的不足之处,最后将会介绍一下关于递归的优化策略,并举例实现。


文章目录

  • 【算法】递归:递归优化之尾递归
    • 一. 递归法求斐波那契数列
    • 二. 递归法的不足
    • 三. 递归优化——尾递归
    • 四. 总结尾递归书写方法
    • 五. 总结

一. 递归法求斐波那契数列

  1. 问题呈现

    写一个函数,输入 n ,求斐波那契(Fibonacci)数列的第 n 项(即 F(N))。

    斐波那契数列的定义如下:F(1) = 1, F(2) = 1,F(N) = F(N - 1) + F(N - 2), 其中 N > 1。

  2. 方法复习:通过分析问题,回答如何分解问题成为相似子问题,如何解决子问题,如何合并子问题的解解决原问题,以及递归终止条件又是什么。

  3. 题目分析:这是一道较为直白的递归问题,首先在定义中我们就知道将问题分解,由F(N) = F(N - 1) + F(N - 2)可知,我们可以将问题分解为F(N-1) 与 F(N-2) 的求和,而要解决该问题,可以通过递归来不断解决,在合并时,我们通过加法完成对子问题的合并解决原问题,最后的终止条件就是在N == 1 || N == 2 时,从递推转推回归。

  4. 经过分析我们可以写出代码:

    int Fibonacci_sequence(int n) {// 终止条件if (n == 1 || n == 2)return 1;else// 分解 解决子问题 合并子问题return Fibonacci_sequence(n - 1) + Fibonacci_sequence(n - 2);
    }
    

二. 递归法的不足

在上述篇幅我们解决了斐波那契额的问题,可以当我们把 n 输入到 50 时,我们就会发现,我们的程序仿佛已经死掉了,其实这主要的原因是在于在递的过程中,因为上一步计算需要先完成下一步计算才可以,所以每个函数在函数栈帧中会占用相应的内存,并且不被释放,知道回归合并为止。比如在此处,输入 n = 50 后,会有大量的内存遗留,甚至导致栈溢出,出现假死。不仅如此,还会出现像重复计算,在函数栈帧的创建与销毁需要时间与空间等问题

总体来说,主要问题有:

  • 递归使用函数栈帧模拟循环的过程,栈帧的开辟和销毁都需要时间;

  • 在递归使用函数栈帧的过程中,如果可能无法及时的释放内存而导致占用内容空间,甚至还会导致栈溢出;

  • 可能会出现大量的重复计算,提高时间复杂度

三. 递归优化——尾递归

所谓的尾递归,我们先给出概念,就是当编译器检测到尾递归时,覆盖当前栈帧,而不是去建立一个新的栈帧,这样只需要占用一个函数栈帧空间,防止了内存的大量浪费。

为了解尾递归,我们需要用尾调用入手,掌握其基本原理,再运用于尾调用。

  1. 尾调用

    函数最后一步调用另外一个函数。

    如以下代码块就是一个典型的尾调用,在调用 test_3() 后,最后一步调用另外一个函数 test_2(),同时覆盖 test_3() 建立的栈帧,以此类推,调用的栈帧永远只有 1 条, 节省了空间。

    void test_1() {printf("test====>1\n");
    }
    void test_2() {printf("test====>2\n");return test_1();
    }
    void test_3() {printf("test====>3\n");return test_2();
    }
    

    还需要区分一下情况不是尾调用,首先 test_5( ) 是一个赋值操作,test_6( ) 是是需要保留我们的函数栈帧的,test_7( ) 并未返回是一个使程序崩溃的。

    int test_4() {return 1;
    }
    int test_5() {int ret = test_4();return ret;
    }int test_6() {return test_4() + 1;
    }void test_7(){test_7();
    }
    
  2. 尾调用优化

    函数最后一步,不需要保留函数帧,其中调用位置与内部变量不需要再被用到。只需要将最后一个函数执行之前return即可。

    void f(){.........return g();
    }
    
  3. 尾递归

    递归就是自己调用自己,尾递归就是自己在最后一步调用自己

    根据尾调用与递归的结合,我们可以使得递归不再出现占用过多内存的问题,防止栈溢出的发生,起到了优化的作用。

    我们要如何做到实现尾调用呢,首先要求函数其中调用位置与内部变量不需要再被用到,我们可以把这些中间量放到参数列表中,其次要求在最后一步调用自己,我们在像递归一样调用即可。

  4. 尾递归解决斐波那契问题

    解决斐波那契问题,n表示递归次数,终止条件是 n=1 或者 n=2 的情况,此时返回 F( n -1) + F( n -2 ) …… F( 2 ) + F ( 1 ) 即可,否则,我们就需要使用尾递归,其中中间值放在参数中,使用 temp1 与 temp2 来代替,其中temp2始终记录每次递归的最后结果。

    int Fibonacci_sequence_better(int n,int temp1,int temp2) {if (n == 1 || n == 2)return temp2;elsereturn Fibonacci_sequence_better(n - 1, temp2, temp1+temp2);
    }
    
  5. 尾递归实例——阶乘

    解决阶乘的尾递归问题,我们也是采取相同的思路,n表示递归次数,在n==1 时为终止条件,我们只需要返回 temp 算好的结果,否则,继续使用尾递归计算,其中参数就存放中间值,这样就可以覆盖栈帧,而不使得原栈帧返回。

    int factorial(int n, int temp) {if (n == 1)return temp;elsereturn factorial(n - 1, temp * n);
    }
    

四. 总结尾递归书写方法

尾递归从本质上来说就是覆盖原有的栈帧空间,防止内存过度占用,而实现的主要条件就是要求函数其中调用位置与内部变量不需要再被用到,为此我们所想出的解决办法就是将这些数据作为中间变量出现,如此就可以防止原函数无法出栈的情况。最后既然是递归,仍旧需要调用自身,完成递归的过程。

为此作者通过上述例题进行进行一个简单的提炼,希望能帮助大家,同时尾递归的运用离不开理解与联系,所以大家必须要着手于实践之中。提炼内容如下:

//n:迭代次数
//temp1与temp2:将需要使用的局部变量通过参数的形式出现
int Fibonacci_sequence_better(int n,int temp1,int temp2) {//终止条件if (n == 1 || n == 2)// temp2 表示累计递归结果return temp2;else//递归调用自身return Fibonacci_sequence_better(n - 1, temp2, temp1+temp2);
}

五. 总结

在介绍递归的文章中,作者写了两篇,其中介绍了自己如何去解决递归问题,然后在本篇中完成了对递归优化的拓展。博主认为本篇所讨论的只是扩展内容,不需过度深究,了解即可,更重要的是解决递归问题的方法。

最后对两篇文章进行概括:

  • 分析解决递归问题:分解问题、解决子问题、合并子问题得到原问题答案、终止条件;
  • 尾递归优化:将局部变量转换为中间参数,在最后调用自身;

递归问题的理解以及运用是需要练习的,所以在掌握理论知识后,仅需要进行大量练习,这样可以巩固所学,同时还可以有自己更新的理解。


补充:

  1. 代码将会放到: https://gitee.com/liu-hongtao-1/c–c–review.git ,欢迎查看!
  2. 欢迎各位点赞、评论、收藏与关注,大家的支持是我更新的动力,我会继续不断地分享更多的知识!
    所以在掌握理论知识后,仅需要进行大量练习,这样可以巩固所学,同时还可以有自己更新的理解。

【算法】递归:递归优化之尾递归相关推荐

  1. foreach jdk8 递归_[Java 8] (8) Lambda表达式对递归的优化(上) - 使用尾递归 .

    递归优化 很多算法都依赖于递归,典型的比如分治法(Divide-and-Conquer).但是普通的递归算法在处理规模较大的问题时,常常会出现StackOverflowError.处理这个问题,我们可 ...

  2. 递归 算法思路和优化和简单实现: 黑白子交换

    原文链接: 递归 算法思路和优化和简单实现: 黑白子交换 上一篇: fabric 绘制心形线 下一篇: 解决steam生化奇兵无限启动报错: steam 服务器 过于繁忙... [已解决] 题目和原解 ...

  3. 2.2基本算法之递归和自调用函数_你为什么学不会递归?读完这篇文章轻松理解递归算法...

    对于很多编程初学者来说,递归算法是学习语言的最大障碍之一.很多人也是半懂不懂,结果学到很深的境地也会因为自己基础不好,导致发展太慢. 可能也有一大部分人知道递归,也能看的懂递归,但在实际做题过程中,却 ...

  4. 史上最全数据结构算法之递归系列学习,建议收藏!

    写在前边 接下来分享的文章是关于递归的,这篇文章不单单分享递归的一切,我觉得更重要的是向每位读者传递一个思想.思想?对的,没错!这篇文章不能说包含递归的边边角角,但是通过自己的理论上的学习和实践,有了 ...

  5. 数据结构与算法之递归系列

    本文来自一个不甘平凡的码农 写在前边 几个月之前就想写这样一篇文章分享给大家,由于自己有心而力不足,没有把真正的学到的东西沉淀下来,所以一直在不断的在自学. 然后又用了一个星期的时间去整理.分类,才有 ...

  6. 常用十大算法 非递归二分查找、分治法、动态规划、贪心算法、回溯算法(骑士周游为例)、KMP、最小生成树算法:Prim、Kruskal、最短路径算法:Dijkstra、Floyd。

    十大算法 学完数据结构该学什么?当然是来巩固算法,下面介绍了十中比较常用的算法,希望能帮到大家. 包括:非递归二分查找.分治法.动态规划.贪心算法.回溯算法(骑士周游为例).KMP.最小生成树算法:P ...

  7. python冒泡排序算法非递归_python 冒泡排序,递归

    今天LeetCode的时候暴力求解233 问题: 给定一个整数 n,计算所有小于等于 n 的非负数中数字1出现的个数. 例如: 给定 n = 13, 返回 6,因为数字1出现在下数中出现:1,10,1 ...

  8. 蓝桥杯 出栈顺序问题引发的思考以及递归的优化(缓存池)

    蓝桥杯 出栈顺序问题引发的思考以及递归的优化(缓存池) 关于递归的优化和思考 在我们IT圈内有句话,普通程序员用迭代,天才程序员用递归.诚然,递归确实能够将许多复杂的问题简化,但是问题来了,由于递归采 ...

  9. 算法之递归(3)- 链表操作

    算法之递归(3)- 链表操作 递归(2)尝试了一个单链表的遍历,同时又分析了如何添加自己的操作,是在递归调用之前,还是在递归调用之后. 今天,打算将问题深入一下,即添加相应的操作在递归的过程中. (免 ...

最新文章

  1. 每日一皮:临近截止日期, 产品经理就这样看着我...
  2. exchange2013警告The maximum number of concurrent connections has exceeded a limit
  3. 【实用】C#测试web服务是否可用
  4. seo 伪原创_胡子哥谈seo优化:那些不被了解的伪原创技巧
  5. 论文浅尝 | Understanding Black-box Predictions via Influence Func
  6. ASP.NET 程序中常用的三十三种代码
  7. Gainlo 面试指南 翻译完成
  8. mysql limit分页_MySQL order by limit 分页数据重复问题
  9. java frame linux_JAVA环境(下) - Android框架简介_Linux编程_Linux公社-Linux系统门户网站...
  10. android 4g获取mac地址,Android手机获取Mac地址的几种方法
  11. java 打印 日历 详细 注解_Java实现按年月打印日历功能【基于Calendar】
  12. java bitset_Java1.8-BitSet源码分析
  13. es文件浏览器访问ftp服务器,es文件浏览器如何ftp服务器
  14. 如何测试网站服务器安全性,怎么检测网站安全性?
  15. 五项python小游戏代码测试
  16. Java使用winrar压缩和解压缩文件
  17. linux小红帽如何封闭端口,安装红帽子Linux的几点注意
  18. 2022工作中遇到问题一
  19. python学习:向Firebird数据库表中插入数据
  20. Online Meetup DevOps World 社区议程和 CFP

热门文章

  1. 找异数java_小米19秋招java开发~ 分享笔试题希望春招有好运
  2. 【树莓派】带你从0到1完成魔镜制作
  3. systemctl 实现开机自启服务
  4. 男孩去铁路好还是学计算机好,男生铁路专业就业前景 铁路学校好就业吗
  5. MySQL数据库进阶版
  6. 合同诈骗案立案标准是怎样的
  7. 初次安装RedFlag5+Oracle
  8. JEWELCAD珠宝手饰设计视频教程 JewelCAD Pro v2.2.2
  9. ubunt查看网卡流量_Linux下查看网卡流量
  10. C语言康威生命游戏,【2020存档】康威生命游戏(CGoL)研究进展