zoukankan      html  css  js  c++  java
  • [转]尾部递归(递归转循环)

    转自:http://www.cnblogs.com/candyming/archive/2011/12/04/2275448.html

    尾部递归是一种编程技巧。递归函数是指一些会在函数内调用自己的函数,如果在递归函数中,递归调用返回的结果总被直接返回,则称为尾部递归。尾部递归的函数有助将算法转化成函数编程语言,而且从编译器角度来说,亦容易优化成为普通循环。这是因为从电脑的基本面来说,所有的循环都是利用重复移跳到代码的开头来实现的。如果有尾部归递,就只需要叠套一个 stack,因为电脑只需要将函数的 parameter 改变再重新跑一次。例如,可以把上一次函数的输出当作下一次的输入。然而,利用尾部递归最主要的目的,是要优化。例如在 Scheme 语言中,明确规定必须针对尾部递归作优化。[1][2]可见尾部递归的作用,是非常依赖于 implementation 的。

    举例说明,最常用作解说递归函数的形式是计算阶乘,如下例

    function factorial(x)
        if (x = 1)
          return 1
        else
          return x * factorial(x-1)
        end if
      end function
    上例中,factorial函数调用了自己,所以这是一递归函数。而且,只在返回指令才调用递归函数,所以这亦是尾部递归(错误,这不是尾部递归,
    因为factorial(x-1)算完后还要乘以x)。以上函数很容易可以转化为循环版本:
    function factorial(x)
        returnvalue = 1
        while (x != 1)
           returnvalue = x * returnvalue
           x = x - 1
        end while
        return returnvalue
      end function
    又例如我们可以替 factorial 设立一个 wrapper function:
    function fact(i, acc)
        if (i == 1)
          return acc
        return fact(i-1, acc*i)
      end function
      
      function factorial(x)
        return fact(x, 1)
      end function
    但是,下列计算斐波那契数列的函数不是尾部递归:
    function fibonnacci(x)
        if (x = 1)
          return 1
        else if (x = 2)
          return 1
        else
          return fibonnacci(x-1) + fibonnacci(x-2)
        end if
      end function
    因为上述函数中,调用了自己两次,亦即一函数的运行会导致两个函数的运行,故此不可能在叫呼的同时退出本来的函数,
    换句话说,调用自己的时后并未到达运行的尾部。
     
    对于递归函数的使用,人们所关心的一个问题是栈空间的增长。确实,随着被调用次数的增加,某些种类的递归函数会线性地
    增加栈空间的使用——不过,有一类函数,即尾部递归函数,不管递归有多深,栈的大小都保持不变。
    函数所做的最后一件事情是一个函数调用(递归的或者非递归的),这被称为尾部调用。使用尾部调用的递归
    称为尾部递归。
    在尾部调用之后除去栈结构的方法称为尾部调用优化。尾部调用优化就是要在尾部进行函数调用时使用下一个
    栈结构覆盖当前的栈结构,同时保持原来的返回地址。
     
    尾部递归和迭代的区别:    这个确实很难区分。迭代中不一定有递归,但递归中一定有迭代,可说递归是迭代的一部分。
        递归是简单的重复调用自己,而迭代则必须有新值出现,而且这个新值是由旧值得来的。
    递归就是一个函数调用自己。尾递归也是递归(一个函数在最末尾的地方调用自己),只不过编译器可能会做优化。一旦优化成功,除了性能的提升之外,还可以避免堆栈溢出的情形。即使无限递归也不会造成溢出。
    迭代应该是指多次计算,每次计算都更加的接近最终结果,因此,计算的次数增多了就能得到足够精确的近似值。迭代一般用循环来做,但没有规定不能用其它方法来做。比如,你可以用递归来做迭代。
    递归是分割子问题  迭代是把用子问题推出大问题,就像算法里面的 搜索 和动态规划一样。
    消除尾部递归(数据结构知识:排序二叉树的内容)
    TreeNode *TreeSearch(TreeNode *root,KeyType target)
     {
         while(root && target != root->entry.key)
             if(target < root->entry.key)
                 root=root->left;
             else
                 root=root->right;
         return root;
     }消除尾部递归
  • 相关阅读:
    2、函数
    二者取其一(初遇)_网络流
    P1879 [USACO06NOV]玉米田Corn Fields
    P2831 愤怒的小鸟
    P2296 寻找道路
    序(不知道是什么时候的模拟题)
    P2243 电路维修
    P1273 有线电视网
    P2613 【模板】有理数取余
    P1373 小a和uim之大逃离
  • 原文地址:https://www.cnblogs.com/lihaozy/p/2799419.html
Copyright © 2011-2022 走看看