C#函数式编程之缓存技术

缓存技术

该节我们将分成两部分来讲解,第一部分为预计算,第二部分则为缓存。缓存这个技术对应从事开发的人员来说是非常熟悉的,从页面缓存到数据库缓存无处不在,而其最重要的特点就是在第一次查询后将数据缓存,在以后的查询过程中就无需重新计算而直接从内存中将结果返回,大大提高了性能,而我们这里的缓存则集中运用在函数上。

预计算

可能一些人并不能立马理解这个词的含义,所以我们就简单的从生活例子出发介绍一下。很多人在工作中一定会这样做事,比如上级吩咐了你一件事,但是这件事的 后半部分要等另一个同事做好之后把对应的材料给你你才能完成。但是我们不可能一直等到那个同事完成了把材料交给我们,我们才去做这件事。而是会将这件事的 前半部分做好,那么剩下的只要那个同事完成并交给我们,我们就直接完成下半部分就可以了。同理这样的思维也可以用在软件开发中,比如下面这个函数:

函数内部是利用a的值计算出c,最后再将c和b相加得出最后的结果。当然这个例子并不能完整的体现预计算的特点,但能够让我们理解预计算是如何实现的。假设这里的

是一个耗时操作,并且实际使用中会出现a的值不变动,但是b的值会经常变动的情况,但是每次调用这个函数都会重新根据a计算出c,那么我们就需要一定的方式改变这个格局,这里我们可以先尝试采用部分应用(这里介绍一个函数式开发的库->PortableFCSLib,可以在NuGet中安装或者到他的github网站上下载:https://github.com/rmoritz/PortableFCSLib):

这里我们的意图是只会计算一次C的值,而不是两次,然后我们看看最终的输出结果:

还是计算了两次的C,理由很简单,因为部分应用仅仅只是利用闭包将参数保存了起来,只有所有参数传递完成后才会调用这个函数,并达不到预计算的效果,所以我们就需要新的方式来完成,下面我们修改DoSomeThing函数:

这里我们可以看到参数只剩下了a,而返回值则变成了函数,这样才执行第一次这个方法之后将会计算出c值,由于闭包的缘故c的值就会被保存。下面我们来看一看如何调用:

最后看看是不是只计算了一次C的值:

这样我们就大功告成了,当然笔者在这里还要再提一下,这些例子仅仅只是为了读者能够快速的理解,在实际的运用中还要读者能够根据情况灵活多变,比如这个函 数接收三个参数,但是前两个不经常变动,但是第三个却经常变动,并且函数的内部是根据前两个参数计算得出一个结果,而返回值需要根据第三个参数和这个计算 后的值得出,那么我们就可以返回但一个参数的函数,而函数本身需要接收两个。

缓存

利用该技术之前我们需要理解几个名词,就是引用透明函数纯度。这两者都是指在我们调用一个函数时,无论任何时候,只要传递的参数一致,返回的结果都应该是一致的。这样的函数我们才能够利用缓存。首先我们先定义一个函数,而这个函数将会是我们后面需要缓存的函数:

然后我们修改函数使之能够进行缓存:

这里我们可以看到我们利用了字典来对这个函数进行了缓存。函数首先从字典中判断是否存在参数a的key,如果存在直接返回计算后的结果,如果不存在则计算 该结果,并保存到字段中,这样我们就实现了一个简单的缓存。下面我们来看看最终的结果,是不是确实使用了缓存:

当传递参数10的时候,因为缓存中没有所以进行了缓存,参数5也是一样,而下一行再次计算10的时候就没有进行计算而是直接从字典中返回了对应的结果。但是上面这种方式还存在一个问题,如果存在多个函数都需要缓存,则这个类会存在多个字段类型的字段(一些人可能会问为什么不能共享一个字典,这样你就要在key的命名上花费一定的功夫,而且很容易造成重复),那么我们就需要一种能够不污染类的方式来进行缓存,这里我们先介绍如何使用FCSLib中的Memoizer实现内部缓存:

这里我们可以看到Memoizer公开了一个GetMemory的静态方式用来获取对应的缓存对象,然后利用这个返回的对象我们就可以进行缓存,最终的效果跟之前的是一样的,我们可以看看最后控制台输出的结果:

如果你不知道该为这个函数起什么名字,我们可以利用反射来获取这个函数的全称,比如下面这个修改之后的DoSomeThing函数就是利用了这个方式:

当然除了手动修改函数的方式,我们也可以采用自动化来使没有利用缓存的函数使用缓存技术,下面我们来写一个函数来实现这个功能:

我们可以看到红色框住的部分,其实就是利用了闭包,在这个函数之上又嵌套了一层函数,这样我们就能够进行缓存了,只有在缓存中不存在时才调用函数求值,但 是面对多个参数的情况,上面这些无法正常缓存了。那么我们就需要使用深度缓存,而所谓的深度缓存就是利用字典套字典来进行保存的,比如下面这个函数,需要 传递两个参数,那么对应的缓存就是:

然后就是Main中进行调用:

最后我们可以看到下面的这个结果,第一次调用sfunc(10,5)时建立了缓存,再第二次传递同样的参数调用后可以看到控制台并没有输出对应的字符串:

当然这样字典的嵌套在参数很多的情况下,会显得很复杂,并且也会消耗很多内存。但是当前也没有非常好的解决方案,下面我们还可以利用之前写的Cache函数来实现上面这种多个参数的缓存:

重点是我们红色框住的那部分,具体的嵌套就是第一个字典的key是第一个参数,value就是下个函数的引用,当然这个函数是经过Cache包装之后的,那么自然在调用value的函数之后自然也起到了缓存作用。

时间: 2024-10-11 03:19:51

C#函数式编程之缓存技术的相关文章

函数式编程思想:耦合和组合,第1部分

总是在某种特定抽象(比如说面向对象)中进行编码工作,这使得很难看清楚何时这一抽象会把你引向一种并非最好的解决方案上.作为这一系列的两篇文章中的头 一篇,本文探讨了用于代码重用的面向对象编程思想的一些影响,并把它们与一些更函数化的可选方法,比如说组合,进行比较. 面向对象编程通过封装变动部分把代码变成易懂的,函数式编程则是通过最小化变动部分来把代码变成易懂的.--Michael Feathers,Working with Legacy Code一书的作者,经由Twitter 每天都以某种特定的抽象

JS函数式编程【译】4.在Javascript中实现函数式编程的技术

?? Functional Programming in Javascript 主目录上一章 建立函数式编程环境 第四章 在Javascript中实现函数式编程的技术 扶好你的帽子,我们现在要真正进入函数式的思想了. 这章我们继续下面的内容: 把所有的核心概念放到一个集中的范式里 探索函数式编程之美 一步步跟踪函数式模式相互交织的逻辑 我们将贯穿整章建立一个简单的应用做一些很酷的事情 你可能已经注意到,在上一章我们介绍Javascript的函数式库的时候引入了一些概念, 而不是在第二章<函数式编

用函数式编程技术编写优美的 JavaScript

用函数式编程技术编写优美的 JavaScript_ibm作者: 字体:[增加 减小] 类型:转载函数式编程语言在学术领域已经存在相当长一段时间了,但是从历史上看,它们没有丰富的工具和库可供使用.随着 .NET 平台上的 Haskell 的出现,函数式编程变得更加流行.一些传统的编程语言,例如 C++ 和 JavaScript,引入了由函数式编程提供的一些构造和特性.在许多情况下,JavaScript 的重复代码导致了一些拙劣的编码.如果使用函数式编程,就可以避免这些问题.此外,可以利用函数式编程

python - 技术提高要点之一,函数式编程,性能,测试和编码规范

摘自:http://www.cnblogs.com/kaituorensheng/p/4516983.html 函数式编程 命令式的编程风格已经成为事实上的标准.命令式编程的程序是由一些描述状态转变的语句组成.虽然有时候这种编程方式十分的有效,但有时也不尽如此(比如复杂性)--而且,相对于声明式编程方式,它可能会显得不是很直观. 如果你不明白我究竟是在说什么,这很正常.这里有一些文章能让你脑袋开窍.但你要注意,这些文章有点像<骇客帝国>里的红色药丸--一旦你尝试过了函数式编程,你就永远不会回头

深入浅出 Python 函数式编程

1.函数式编程的定义与由来 如果程序中的函数仅接受输入并产生输出,即输出只依赖于输入,数据不可变,避免保存程序状态,那么就称为函数式编程(Functional Programming,简称FP,又称泛函编程). 这种风格也称声明式编程(Declarative Programming),与之相对的是指令式编程(Imperative Programming),后者中的对象会不断修改自身状态.函数式编程强调程序的执行结果比执行过程更重要,倡导利用若干简单的执行单元让计算结果不断渐进,逐层推导复杂的运算

函数式编程扫盲篇

1. 概论 在过去的近十年的时间里,面向对象编程大行其道.以至于在大学的教育里,老师也只会教给我们两种编程模型,面向过程和面向对象. 孰不知,在面向对象产生之前,在面向对象思想产生之前,函数式编程已经有了数十年的历史. 那么,接下来,就让我们回顾这个古老又现代的编程模型,让我们看看究竟是什么魔力将这个概念,将这个古老的概念,在21世纪的今天再次拉入了我们的视野. 2. 什么是函数式编程 在维基百科中,已经对函数式编程有了很详细的介绍. 那我们就来摘取一下Wiki上对Functional Prog

C#函数式编程之惰性求值

惰性求值 在开始介绍今天要讲的知识之前,我们想要理解严格求值策略和非严格求值策略之间的区别,这样我们才能够深有体会的明白为什么需要利用这个技术.首先需要说明的是C#语言小部分采用了非严格求值策略,大部分还是严格求值策略.首先我们先演示非严格求值策略的情况,我们先在控制台项目中写一个DoOneThing方法. 然后在Main方法中写入下面这串代码: 然后我们运行程序,会发现DoOneThing方法并没有执行.当然这看起来也很正常,因为这是或,并且第一个已经是true了.整个表达式就是true了,自

C#函数式编程

转载:http://www.admin10000.com/document/9216.html 提起函数式编程,大家一定想到的是语法高度灵活和动态的LISP,Haskell这样古老的函数式语言,往近了说ruby,javascript,F#也是函数式编程的流行语言.然而自从.net支持了lambda表达式,C#虽然作为一种指令式程序设计语言,在函数式编程方面也毫不逊色.我们在使用c#编写代码的过程中,有意无意的都会使用高阶函数,组合函数,纯函数缓存等思想,连表达式树这样的idea也来自函数式编程思

使用C#进行函数式编程

提起函数式编程,大家一定想到的是语法高度灵活和动态的LISP,Haskell这样古老的函数式语言,往近了说ruby,javascript,F#也是函数式编程的流行语言.然而.net自从支持了lambda表达式,C#虽然作为一种指令式程序设计语言,在函数性编程方面也毫不逊色.我们在使用c#编写代码的过程中,有意无意的都会使用高阶函数,组合函数,纯函数缓存等思想,连表达式树这样的idea也来自函数式编程思想.所以接下来我们把常用的函数式编程场景做个总结,有利于我们在程序设计过程中灵活应用这些技术,拓