java递归转非递归_递归和非递归算法的转化(Java)

递归函数具有很好的可读性和可维护性,但大部分情况下程序效率不如非递归函数,所以在程序设计中一般喜欢先用递归解决问题,在保证方法正确的前提下,如果需要则再转换为非递归函数以提高效率。

函数调用时,需要在栈中分配新的帧,将返回地址、调用参数和局部变量入栈。所以递归调用越深,占用的栈空间越多。如果层数过深,肯定会导致栈溢出,这也是消除递归的必要性之一

递归函数是编译器自动实现入栈和出栈,而非递归则手动设置栈的各种操作(可以用来计算空间复杂度)

本质:递归的本质是压栈,循环中加上栈操作,其实和递归没有本质区别,所以,所有递归都可以用非递归的方式实现。

递归函数:尾递归和非尾递归函数

1.尾递归

定义:函数返回前的最后一个操作是调用递归函数,也即调用自己,不存在其他的操作运算

特点:① 调用自身函数(Self-called); ②空间仅占用常量栈空间(Stack Space)。

推理:正确实现尾递归的语言不存在栈溢出问题

举例:

public int FibonacciTailRecur(int n, int a, int b) {

if (n == 0) return a;

return FibonacciTailRecur(n-1, b, a+b);

}

2.非尾递归

非递归是最常见的递归形式,因为中间涉及各种操作是工程必然的

举例:

public int FibonacciRecur(int n) {

if (0==n) return 0;

if (1==n) return 1;

return FibonacciRecur(n-1)+FibonacciRecur(n-2);

}

模型:递归转化为非尾递归的模型如下

建立一个新的栈实例

把原始内容压入新建栈;

while (栈不为空 || other condition){

出栈,将栈顶元素赋给s;

if (s是要找的结果){

返回;

}

else {

寻找到s的相关状态s1;

将s1进栈

}

}

工程问题

虽然将递归函数转换为迭代函数可以提高程序效率,但是转换后的迭代函数往往可读性差,难以理解,不易维护。所以只有在特殊情况下,比如对栈空间有严格要求的嵌入式系统,才需要转换递归函数。大部分情况下,递归并不会成为系统的性能瓶颈,一个代码简单易读的递归函数常常比迭代函数更易维护。

Reference

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值