时间:2021-05-28
一个函数输出当做另一个函数输入。有时候一个复杂问题,我们拆分成很多个步骤函数,这些函数组合起来调用解决一个复杂问题。
在C#中不支持函数组合,但可以直接像这样调用B(A(n)),这也是函数组合,但这不利于阅读,人们习惯从左往右阅读,而不是相反的方向。通过创建扩展方法可以任何组合两个函数,像下面这样
Func<A,C> Compose<A,B,C>(this Func<A.B> f ,Func<B,C> g)=>(n)=>g(f(n))上述代码为泛型委托Func<a,b>创建了一个扩展Compose的扩展方法,以泛型委托Func<b,c>为输入参数,返回组合后的函数Func<a,c>。创建一个高阶函数Compose把不利于阅读的隐藏起来。
在F#中就非常方便的使用函数组合。举个例子,将一个列表中数字增加4再乘以3,构建这两个步骤的函数(当然利用C#linq或F#map可以直接(x+4)*3,这里主要演示两个功能函数如何组合起来)。
letadd4x=x+4letmulitply3x=x*3letlist=[0..10]letnewList=List.map(funx->mulitply3(add4(x)))listletnewList2=list|>List.map(add4>>mulitply3在F#中使用>>中缀运算符来使函数组合可以从左到右阅读,更加精炼、简洁。
闭包可以让函数访问其所在的外部函数中的参数和变量,即使在其外部函数被返回之后。在js中经常会出现闭包的场景,在C#和F#中,编译器使用闭包来增加和扩展变量的范围。
C#在.NET2.0后引入闭包。在lambda和匿名方法中得到充分的使用。像下面的匿名函数引用变量a,访问和管理变量a的状态。如果不用闭包,就需要额外创建一个类函数来调用。
strings="freevariable";Func<string,string>lambda=value=>a+""+value;以下载图片更新窗体PictureBox控件为例:
void UpdateImage(string url){ System.Windows.Forms.PictureBox picbox = this.pictureBox1; var client = new WebClient(); client.DownloadDataCompleted += (o, e) => { if (picbox != null) { using (var ms = new MemoryStream(e.Result)) { picbox.Image = Image.FromStream(ms); } } }; client.DownloadDataAsync(new Uri(url)); //picbox = null;}因为是异步下载,UPdateImage方法返回后,图片还未下载完成,但picbox变量仍然可以使用。这就是变量捕获。lambda表达式捕获了局部变量image,因此它仍停留在作用域中。但捕获的变量值是在运行时确定的,而不是在捕获时,最后一句如果放开,将不能更新窗体。运行时picbox为null了,在F#中不存在null的概念,所以也不会出现此类错误。
多线程环境中的闭包使用。猜测下面的代码运行结果如何?
for (int i = 1; i < 10; i++){ Task.Factory.StartNew(()=>Console.WriteLine("{0}-{1}", Thread.CurrentThread.ManagedThreadId,i));}不会按期望的那样打印1-9,因为他们共享变量i,调用时i的值可能已经被循环修改了。印证上面说的捕获的变量值是在运行时确定的。
这种情况就很难搞,给并行编程带来了头疼的问题,变量可变,这不废话吗,变量不会变就不叫变量了。在C#中解决此类问题的一个方法就是为每个任务创建创建和捕获一个新的临时变量,这样它就能保留捕获时的值。在F#中不存在这个问题,它的For循环每次创建一个新的不可变值。
一些函数会频繁的使用相同的参数去调用。我们可以将用相同的参数调用函数的结果存储起来,以便下次调用直接返回结果。例如对图片每个像素做处理,一张图片可能相同像素的会有很多,通过缓存可以直接返回上次计算结果。
//简单的函数缓存public static Func<T, R> Memoize<T, R>(Func<T, R> func) where T : IComparable { Dictionary<T, R> cache = new Dictionary<T, R>(); return arg => { if (cache.ContainsKey(arg)) return cache[arg]; return (cache[arg] = func(arg)); };}// 线程安全的函数缓存public static Func<T, R> MemoizeThreadSafe<T, R>(Func<T, R> func) where T : IComparable{ ConcurrentDictionary<T, R> cache = new ConcurrentDictionary<T, R>(); return arg => cache.GetOrAdd(arg, a => func(a));}// 利用延迟提高性能的函数缓存public static Func<T, R> MemoizeLazyThreadSafe<T, R>(Func<T, R> func) where T : IComparable{ ConcurrentDictionary<T, Lazy<R>> cache = new ConcurrentDictionary<T, Lazy<R>>(); return arg => cache.GetOrAdd(arg, a => new Lazy<R>(() => func(a))).Value;}上述示例代码中有三个版本的函数记忆化。调用像下面这样
public static string Greeting(string name){ return $"Warm greetings {name}, the time is {DateTime.Now.ToString("hh:mm:ss")}";}public static void RunDemoMemoization(){ var greetingMemoize = Memoize<string, string>(Greeting); Console.WriteLine(greetingMemoize("Richard")); Console.WriteLine(greetingMemoize("Paul")); Console.WriteLine(greetingMemoize("Richard"));}线程安全字典ConcurrentDictionary可以保证只向集合里添加一个相同值,但函数求值可能会被执行多次,所以利用.NET4之后的延迟对象加载技术。在真正需要使用对象时候才去实例化(通过访问延迟对象的Value属性),而且是线程安全的。
到此这篇关于.NET并发编程之函数闭包的文章就介绍到这了,更多相关.NET函数闭包内容请搜索以前的文章或继续浏览下面的相关文章希望大家以后多多支持!
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
Python基础教程之闭包的使用方法前言:闭包(closure)是函数式编程的重要的语法结构。函数式编程是一种编程范式(而面向过程编程和面向对象编程也都是编程范
闭包的由来形成闭包有一些值得总结的非必要条件:1、嵌套定义的函数。2、匿名函数。3、将函数作为参数或者返回值。4、在.NET中,可以通过匿名委托形成闭包:函数可
闭包(Closures)是独立的函数代码块,能在代码中传递及使用。Swift中的闭包与C和Objective-C中的代码块及其它编程语言中的匿名函数相似。闭包可
Go语言最主要的特性复制代码代码如下:自动垃圾回收更丰富的内置类型函数多返回值错误处理匿名函数和闭包类型和接口并发编程反射语言交互性高性能/高效开发安装安装说明
JavaScript是函数式编程语言,支持高阶函数和闭包。函数式编程非常强大,可以写出非常简洁的代码。例如Array的map()和filter()方法:'use