C#函数式编程中的惰性求值详解

作者:junjie 时间:2022-01-27 03:07:29 

惰性求值

      在开始介绍今天要讲的知识之前,我们想要理解严格求值策略和非严格求值策略之间的区别,这样我们才能够深有体会的明白为什么需要利用这个技术。首先需要说明的是C#语言小部分采用了非严格求值策略,大部分还是严格求值策略。首先我们先演示非严格求值策略的情况,我们先在控制台项目中写一个DoOneThing方法。

C#函数式编程中的惰性求值详解

然后在Main方法中写入下面这串代码:

C#函数式编程中的惰性求值详解

       然后我们运行程序,会发现DoOneThing方法并没有执行。当然这看起来也很正常,因为这是或,并且第一个已经是true了。整个表达式就是true了,自然第二个就无需求值了。但是这恰恰就是非严格求值的策略,如果是严格求值策略的话整个表达式都会计算。接着就是严格求值策略的情况了,这个相信很多人都会立马明白,首先我们需要再写一个DoSomeThing方法:

C#函数式编程中的惰性求值详解

接着修改Main方法:

C#函数式编程中的惰性求值详解

执行之后我们可以看到如下的结果:

C#函数式编程中的惰性求值详解

       但是我们可以清楚的看到a的值是false,根本不会使用b值,但是传递参数的时候已经将DoOneThing方法执行并赋值给b,假设这个方法是一个非常耗时的操作。那么我们就会白白浪费掉这段时间,最后求得的值也没有使用的到。而这正是严格求值策略,而今天的主要目标就是改变这种情况,能够在我们确定需要某个值的时候才计算。下面我们就可以开始改造这个方法,让其能够支持惰性求值。首先我们修改DoSomeThing方法:

C#函数式编程中的惰性求值详解

      这里我们将参数类型都改成了函数,这样将要传递进来的参数都改变成函数。只有在我们需要的时候才执行求值,否则是不会运行的,对应的Main方法中我们需要按照如下方式修改:

C#函数式编程中的惰性求值详解

      这里我们并不需要把DoOneThing方法的返回类型改掉,如果这样的话。在现有项目上使用函数式编程就会显得太麻烦了。这里我们仅仅只需要利用匿名函数就可以办到了,下面我们可以看最后的执行效果:

C#函数式编程中的惰性求值详解

       DoOneThing方法并没有执行,因为DoSomeThing中根本没有确定使用这个变量,这样我们就能够节省下这部分计算的时间,但是事实上我们还没有结束,实际的开发中我们可能需要多次使用这个值,比如下面我们修改DoSomeThing方法:

C#函数式编程中的惰性求值详解

并且在Main方法中调用DoSomeThing方法时将第一个参数改成true,然后执行我们就可以看到下面的输出结果:

C#函数式编程中的惰性求值详解

 DoOneThing方法被执行了两次,当然我们可以利用局部变量保存,可能你会这么写:

C#函数式编程中的惰性求值详解

      如果这么写,那么我们的惰性求值就没有任何意义了,因为一进入这个方法就执行了这个方法,跟传递参数时直接将运算后的结果赋值给b没有任何区别了。当然也有其他一些技巧可以避免,但是这些技巧可不是下面要讲的内容,我们可以将其封装起来,比如我们可以写个LazyS<T>类:

C#函数式编程中的惰性求值详解

      我们可以看到在构造方法部分我们将对应的函数作为参数接收并保存到function中,只有再调用Value时候会执行该函数并将值保存,并且在下次调用时,如果已经求值过则直接返回缓存过的值,这样就能够避免重复的执行了,对应的我们还要修改DoSomeThing方法和Main方法:

C#函数式编程中的惰性求值详解

C#函数式编程中的惰性求值详解

最终执行后我们可以看到仅执行了一次DoOneThing方法:

C#函数式编程中的惰性求值详解

       一些读者可能为问为什么类名不要Lazy而是加个S,因为.net中已经为我们包含了Lazy<T>类,相信很多人基本上从没有用过。只知道Func和Action的存在,下面我们修改我们的代码直接利用自带的:

C#函数式编程中的惰性求值详解

最终的结果之前的是一摸一样,当然系统自带的Lazy功能更多,并且支持多线程。

 就到这里为止吧,周五了大家已经按耐不住了,写了太多可能就没有心思往下看了,所以将这些全部一个一个拆分开来细讲。

标签:C#,函数式编程,惰性求值
0
投稿

猜你喜欢

  • C# 中的 is 真的是越来越强大越来越语义化(推荐)

    2021-06-24 11:55:12
  • c#使用Socket发送HTTP/HTTPS请求的实现代码

    2023-10-12 07:10:00
  • ref与out之间的区别深入解析

    2023-08-05 23:45:28
  • Java编程中使用throw关键字抛出异常的用法简介

    2023-08-27 17:04:19
  • Maven如何修改打包文件名称

    2022-09-02 21:29:52
  • SpringDataJpa like查询无效的解决

    2021-12-02 11:32:40
  • 深入理解MyBatis中的一级缓存与二级缓存

    2022-05-25 09:41:34
  • IDEA中的.iml文件和.idea文件夹

    2023-11-23 11:47:19
  • Android高级xml布局之输入框EditText设计

    2022-09-24 14:25:56
  • Android view自定义实现动态进度条

    2023-09-16 00:40:38
  • Android开发返回键明暗点击效果的实例代码

    2022-06-08 06:39:27
  • mybatis 实现多层级collection嵌套

    2022-07-19 21:33:18
  • 浅谈Java中IO和NIO的本质和区别

    2023-11-01 01:44:08
  • C语言数据类型转换实例代码

    2023-12-04 11:48:36
  • Java面试题冲刺第十六天--消息队列

    2022-08-08 09:07:04
  • Java设计模式之命令模式_动力节点Java学院整理

    2023-12-11 13:32:03
  • Java可重入锁的实现原理与应用场景

    2023-03-27 20:21:54
  • Android App中自定义View视图的实例教程

    2021-08-16 16:54:35
  • 一篇文章教会你用Unity制作网格地图生成组件

    2023-09-05 04:44:36
  • 如何使用java修改文件所有者及其权限

    2023-11-16 09:35:53
  • asp之家 软件编程 m.aspxhome.com