C#函数式编程中的递归调用之尾递归详解

 更新时间:2015年01月28日 09:20:52   投稿:junjie  
这篇文章主要介绍了C#函数式编程中的递归调用详解,本文讲解了什么是尾递归、尾递归的多种方式、尾递归的代码实例等内容,需要的朋友可以参考下

关于递归相信大家已经熟悉的不能再熟悉了,所以笔者在这里就不多费口舌,不懂的读者们可以在博客园中找到很多与之相关的博客。下面我们直接切入正题,开始介绍尾递归。

尾递归

普通递归和尾递归如果仅仅只是从代码的角度出发来看,我们可能发现不了他的特点,所以笔者利用两张堆栈上的图来展示具体的差距在哪,首先我们来看看普通的递归调用的情况,如下图1.1所示:

假设这里执行的函数是Func1,并且Func1中通过递归调用了自己,那么我们可以看到栈上在每次调用Func1的时候都会重新将函数返回地址等其他参数放入栈中,在递归次数较少的情况下,这样是不会有问题的。但如果递归调用次数达到一定的数量级,则会将栈空间消耗光。因此,就提出了尾递归。而尾递归的栈图如1.2所示:

一样还是递归,但是每次执行自身的时候并不会在栈空间中申请新的空间,类似于for循环的效果,面对递归次数很多的情况下也不会出现什么问题。但是新的问题就出来了,在C#中编译器不会做到这一步优化,而是在jit编译器执行时才会进行优化。并且只有64位才进行优化。在语言的层面上我们也要遵守一定的原则,才能让编译器知道去优化。当然有些喜欢看博客的人可能早就知道尾递归就是在最后return的时候调用自身。我们可以通过一串示意代码来看尾调用:

复制代码 代码如下:

int Func1()
{
  return Func1();
}

当然上面这串代码会形成一个死循环,因为这里我们没有基线条件。下面我们举一个例子:

这个函数想必应该会比较熟悉,就是计算阶乘的。但是我们可以发现函数sunfc最后的返回语句并不是直接调用函数本身,而是x*sfunc(x -1),恰恰就是因为前面这个x*就会导致编译器无法优化,从而只能采用普通的递归调用的方式去执行,那么我们就需要利用一些模式去改变,首先我们先介绍的是“累加器传递模式”,可能名字比较悬乎,其实就是将当前的计算结果传递给下一次调用函数中,这样当到达基线条件后直接根据上次计算的结果算出最终结果返回即可,如果将上面的代码采用这个模式就是下面这个样子:

采用这个模式之后我们就变回了尾递归了,当执行到基线条件时,直接返回y的值即可。根本不需要回溯到以前。除了利用这种模式,我们还可以利用一种“后继传递模式”,跟累加器传递模式一样也需要修改函数签名,增加一个参数,我们继续修改上面这串代码:

 

相比累加器传递模式,这种方式比较难理解,其实sfunc在到达基线条件时y就等同于下面这个lambda表达式:a => a*4*3*2,然后就是调用y(1)就直接计算最终的结果了。在简单点就是y这个函数被包装了了好几层,比如上面这段函数执行结束时y的调用顺序:

a为1传递给y(2 * a),结果就是y(2)。

a为2传递给y(3 * a),结果就是y(6)。

a为6传递给y(4 * a),结果就是y(24)。

a为24传递给x => x,输出24。


如果还是不理解只能下断点,调试自己琢磨琢磨了,实在不懂的可以Q问。

 

在满足必要的经济的条件下,研究更加高深的技术.满足自己的野心

相关文章

  • 利用C#实现修改图片透明度功能

    利用C#实现修改图片透明度功能

    这篇文章主要为大家详细介绍了如何利用C#实现修改图片透明度功能,文中的示例代码讲解详细,对我们学习C#有一定的帮助,感兴趣的小伙伴可以了解一下
    2022-12-12
  • C#缩略图多路径多格式保存的实例

    C#缩略图多路径多格式保存的实例

    这篇文章介绍了C#缩略图多路径多格式保存的实例,有需要的朋友可以参考一下
    2013-07-07
  • C语言使用getch()读取方向键

    C语言使用getch()读取方向键

    getch()是编程中所用的函数,这个函数是一个不回显函数,当用户按下某个字符时,函数自动读取,无需按回车,有的C语言命令行程序会用到此函数做游戏,但是这个函数并非标准函数,要注意移植性
    2021-07-07
  • 详解C#中委托,事件与回调函数讲解

    详解C#中委托,事件与回调函数讲解

    这篇文章主要介绍了详解C#中委托,事件与回调函数讲解,小编觉得挺不错的,现在就分享给大家,也给大家做个参考。
    2016-12-12
  • 基于.net中突破每客户端两个http连接限制的详细介绍

    基于.net中突破每客户端两个http连接限制的详细介绍

    本篇文章是对.net中突破每客户端两个http连接限制进行了详细的分析介绍,需要的朋友参考下
    2013-05-05
  • SuperSocket封装成C#类库的步骤

    SuperSocket封装成C#类库的步骤

    这篇文章主要介绍了SuperSocket封装成C#类库的步骤,帮助大家更好的理解和使用c#,感兴趣的朋友可以了解下
    2021-01-01
  • C#使用SqlConnection连接到SQL Server的代码示例

    C#使用SqlConnection连接到SQL Server的代码示例

    这篇文章主要介绍了C#使用SqlConnection连接到SQL Server的代码示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-03-03
  • C#使用LINQ查询操作符实例代码(二)

    C#使用LINQ查询操作符实例代码(二)

    这篇文章介绍了C#使用LINQ查询操作符的方法,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-06-06
  • C#简易图片格式转换器实现方法

    C#简易图片格式转换器实现方法

    这篇文章主要介绍了C#简易图片格式转换器实现方法,涉及C#基于WinForm操作图片的相关技巧,具有一定参考借鉴价值,需要的朋友可以参考下
    2015-11-11
  • c#基础系列之System.String的深入理解

    c#基础系列之System.String的深入理解

    这篇文章主要给大家介绍了关于c#基础系列之System.String的深入理解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2018-09-09

最新评论