递归代码都可以转为非递归吗 ?

开发 前端
我们知道将递归调用全部展开后其实会形成一棵树,把递归转为非递归无非就是在遍历这棵树,那么遍历树就有很多技术了。

先说答案,这是肯定的,所有递归代码都可以转为非递归代码。之所以所有的递归都能转为迭代算法是因为递归借助函数调用,函数调用本身就是基于调用栈这种结构实现的,只不过这一切都是自动完成的,我们当然也可以用代码手动模拟出来。

我们知道将递归调用全部展开后其实会形成一棵树,把递归转为非递归无非就是在遍历这棵树,那么遍历树就有很多技术了,基于栈的深度优先遍历Depth-first traversal,或者基于队列的广度优先遍历breadth-first traversal都是可以的:

说会递归转为非递归这个话题,更理论一些的解释是这样的,不管是递归还是非递归,这两者都是图灵完备的,既然是图灵完备,那么它们在表达能力上就是等价的,不存在谁不能转为谁的问题。关于图灵完备参考这篇《计算机科学中那些有趣的事实》。

只不过这存在一个难易程度的问题。大家都知道尾递归最容易转为非递归的迭代形式,本质上是这棵树不是多叉的而是单叉的,单叉的不就退化成链表了嘛,遍历链表当然是简单的,但如果是多叉的话问题就没那么简单了,这里最有趣的是不存在一种模板可以让我们直接用套路的形式把递归转为非递归,因此这里存在一个问题:为什么你要把递归转为非递归呢?

因为最终你会发现将递归转为非递归无非就是你自己接手了编译器本来已经替你完成的工作,你会发现自己在手动模拟函数调用。

递归的优势很明显:代码简洁,容易理解和维护,其为人诟病的地方在于执行效率“可能”没有非递归版本的高,但你最好理解这句话到底在说什么,到底哪里效率就不高了?我们知道函数调用本身并不是免费的,函数调用也是有代价的,这里的代价就在于维护函数调用以及函数返回需要额外执行一些指令,关于这部分的内容可以参考《​​函数调用时底层发生了什么?​​》,同时栈区空间有限,因此如果你的递归调用层级太多的话可能会导致栈溢出,撑爆你的运行时环境以及可能存在重复计算问题(可以利用memory table解决),除此之外除非你有绝对令人信服的理由,否则你不应该试图将递归转为非递归。

责任编辑:武晓燕 来源: 码农的荒岛求生
相关推荐

2023-04-03 09:44:18

2020-12-07 14:58:50

程序员开源开发者

2009-11-12 09:51:11

Visual C++项

2014-09-24 13:04:13

微信企业号

2013-04-08 12:41:35

JavaScriptJS

2022-03-07 10:27:03

Linux开源社区

2023-03-20 18:34:02

营销大促质量保障稳定性

2011-06-17 14:36:50

Linux

2021-09-15 07:40:50

二叉树数据结构算法

2013-07-04 09:46:06

X Phone

2015-07-13 09:26:12

程序员软技能测验

2022-03-11 11:40:26

AI数据技术

2009-11-18 15:05:56

PHP递归数组

2018-12-04 21:38:33

2019-08-09 10:52:58

Linux内核Spark

2023-07-06 14:43:46

数字化转型数据资源

2019-08-09 10:35:54

2012-11-15 11:00:00

2020-11-18 18:58:29

低代码

2020-05-27 07:38:36

尾递归优化递归函数
点赞
收藏

51CTO技术栈公众号