大家都知道递归,尾递归呢?什么又是尾递归优化?

开发 前端
今天,我们来聊聊递归函数。为啥突然想到递归?其实就从电影名字《恐怖游轮》《盗梦空间》想到了。

 今天,我们来聊聊递归函数。为啥突然想到递归?其实就从电影名字《恐怖游轮》《盗梦空间》想到了。

[[327759]]

 

递归是啥?

 

递归函数大家肯定写过,学校上课的时候,估计最开始的例子就是斐波拉契数列了吧。例如:

  1. int Fibonacci(n) { 
  2.     if (n < 2) return n; 
  3.     return Fibonacci(n - 1) + Fibonacci(n - 2); 

递归函数简而言之就是在一个函数中,又“递归”调用自己。在写递归函数的时候,需要注意的地方就是递归函数的结束条件。用递归函数确实能简化很多算法的实现,比如常见的二叉树遍历等。但往往在写递归函数的时候,最容易出现的问题就是所谓的“栈溢出”。

为什么会有“栈溢出”呢?因为函数调用的过程,都要借助“栈”这种存储结构来保存运行时的一些状态,比如函数调用过程中的变量拷贝,函数调用的地址等等。而“栈”往往存储空间是有限的,当超过其存储空间后,就会抛出著名的异常/错误“StackOverflowError”。

我们以一个简单的加法为例,例如:

  1. int sum(int n) { 
  2.     if (n <= 1) return n; 
  3.     return n + sum(n-1); 
  4.  
  5. std::cout << sum(100) << std::endl; 
  6. std::cout << sum(1000000) << std::endl; 

很简答,编译运行后,比较小的数字,能得到正确的答案,当数字扩大后,就会直接发生“segmentation fault”。

尾递归又是啥?

 

我得知这个概念,最开始还是因为很多年前一次面试,面试官问我“你知道什么是尾递归吗?”,我以为是“伪”递归,难道是假的递归???当初我也是懵逼状态(当初面试官忍住没笑也是厉害了)。从“尾”字可看出来即若函数在尾巴的地方递归调用自己。上面的例子写成尾递归,就变成了如下:

  1. int tailsum(int n, int sum) { 
  2.     if (n == 0) return sum
  3.     return tailsum(n-1, sum+n); 

可以试试结果,计算从 1 加到 1000000,仍然是segmentation fault。为什么呢?因为这种写法,本质上还是有多层的函数嵌套调用,中间仍然有压栈、出栈等占用了存储空间(只不过能比前面的方法会省部分空间)。

尾递归优化

 

当你给编译选项开了优化之后,见证奇迹的时刻到了,居然能算出正确结果。如图所示:

 

 

C++ 默认 segmentation fault, 开启编译优化后,能正常计算结果。

 

原因就是因为编译器帮助做了尾递归优化,可以打开汇编代码看看(这里就不展示 C++的了)。后面我用大家比较熟悉的 JVM based 语言 Scala 来阐述这个优化过程。(好像 Java 的编译器没做这方面的优化,至少我实验我本地 JDK8 是没有的,不清楚最新版本的有木有)(scala 本身提供了一个注解帮助编译器强制校验是否能够进行尾递归优化@tailrec)

  1. object TailRecObject { 
  2.  
  3.    def tailSum(n: IntsumInt): Int = { 
  4.         if (n == 0) return sum
  5.         return tailSum(n-1, n+sum); 
  6.    } 
  7.  
  8.    def main(args: Array[String]) { 
  9.       println(tailSum(100, 0)) 
  10.       println(tailSum(1000000, 0)) 
  11.    } 
  12.  

结果如下图所示,默认情况下 scalac 做了尾递归优化,能够正确计算出结果,当通过 -g:notailcalls 编译参数去掉尾递归优化后,就发生了 Exception in thread "main" java.lang.StackOverflowError了。

 

默认启用尾递归优化正常计算结果,禁用尾递归优化则“StackOverflow”。

 

我们来看看生成的字节码有什么不同。

 

包含尾递归优化的字节码,直接 goto 循环。

 

禁用尾递归优化的字节码,方法调用。

 

从上面可以看出,尾递归优化后,变成循环了(前面的 C++ 类似)。

好了,尾递归咱们就了解到这里。个人看法,我们知道有“尾递归”这个点就好了,有时候我们写递归就是为了方便,代码可读性好,如果确实是出于性能考虑,我们可以自己用迭代的方式去实现,不依赖于具体的编译器实现。当然对于像 scala 这样,有一些语法糖能够帮助校验和验证,也是一个不错的选择。但递归转迭代的能力,我们能具备岂不更好。

本文转载自微信公众号「 程序猿石头」,可以通过以上二维码关注。转载本文请联系 程序猿石头公众号。

 

责任编辑:武晓燕 来源: 程序猿石头
相关推荐

2010-09-17 13:01:44

Python

2009-07-22 07:44:00

Scala尾递归

2020-09-30 08:07:46

如何优化尾调用

2021-03-24 10:00:32

Python递归函数Python基础

2022-10-10 08:13:16

递归通用代码

2019-09-16 08:32:59

递归算法编程

2020-07-10 08:15:19

递归算法函数

2022-03-31 08:15:59

递归代码非递归

2019-09-18 10:12:37

递归数据结构

2024-11-29 14:50:45

2012-02-22 14:12:08

算法

2009-11-17 16:53:24

PHP递归算法

2009-06-06 19:20:08

java麻将和牌程序递归

2010-04-02 15:04:14

Oracle递归查询

2013-11-11 10:03:29

递归路由

2020-02-21 16:43:00

C语言编程语言程序员

2018-03-05 08:52:57

2015-07-08 16:06:15

iOS递归锁

2019-03-26 08:15:45

iOS尾调用Objective-C

2021-09-15 07:40:50

二叉树数据结构算法
点赞
收藏

51CTO技术栈公众号