大到可以小说的Y组合子(一)
问:上回乱扯淡了一通,这回该讲正题了吧。
答:OK. 先来列举一些我参考过,并从中受到启发的文章。
(1.)老赵的一篇文章:使用Lambda表达式编写递归函数
(2.)装配脑袋的两篇文章:VS2008亮点:用Lambda表达式进行函数式编程和用Lambda表达式进行函数式编程(续):用C#实现Y组合子
(3.)Y组合子的推导过程(用Scheme推导),这里的“推导”并不是数学意义上上的推导证明,而是说如何一步步引导构想出Y来的,值得一看。也许从某种程度反应出了当年Y是如何被发明的。
[还有一些文章一时半会儿也找不到了,哪天记起来再过来补上]
本文决定用C#来展示列子,因为C#对普通函数和Lambda表达式都支持得非常好,方便进行对比。同时,会展示其它语言与之不同之处(如js,scheme是动态类型语言,因此实现Y简单不少;F#应该算同门师兄弟了,除了语法更间接之外,也有一个不大不小的区别;C++是一个更加静态的语言,实现起来难度更大些,需要一个不大不小的技巧,这些会在后面章节中展示)。
还是从那个古老但有效的例子开始吧——factorial。这是一个普通的递归函数:
//C#
int Fact(int x) { return x == 0 ? 1 : x * Fact(x - 1); }
所谓递归,就是在函数体内调用函数本身,然而Lambda表达式(下文简称Lambda)是一个匿名函数,又能通过什么去调用自己呢?我曾看到网友给出过这样一个啼笑皆非的方案:
//C#
Func<int, int> fact = null;
fact = x => x == 0 ? 1 : x * fact(x - 1);
对应到F#中就是这样的,这就跟前面C#中的函数颇为相似了:
//F#
let rec fact = fun x -> if(x=0) then 1 else x * fact(x-1)
可是这并没有实现所谓的匿名递归,因为你的确为了调用自身,而给自身取了一个名字“fact”,且它被上下文依赖。所以我们来规定来定义一个匿名递归Lambda的“衡量标准”:仍然以factorial函数为例,说我需要这么一个表达式(其中的所有参量不被上下文依赖,即构成闭包),将其代入到以下代码的(▲)处,可以计算得到参数(5)的阶乘值(120)。
//C#伪码
int result = (▲)(5); //120
那不为自身命名,又如何调用之呢?请看下面这两行伪码,其中T?代表某种未知类型:
//C#伪码
T? fact_maker = self => x => x == 0 ? 1 : x * self(x - 1);
int result = fact_maker(fact_maker)(5);
我们期望的是,把这个Lambda通过self参量传递给Lambda自己(就像fact_maker(fact_maker)做的那样),以此来返回如上伪码的下划线部分(即类型为Func<int,int>的fact函数)。但是这里有个矛盾——我们既希望self能接受fact_maker(类型为Func<Tself,Func<int,int>>,其中Tself代表self的类型),又希望self在Lambda内部扮演fact的角色(类型为Func<int,int>),所以,这样是行不通的。既然我们希望fact是self的返回值,不如这样试试:
//C#伪码
Funct<Tself,Func<int,int>> fact_maker =
self => x => x == 0 ? 1 : x * self(self)(x - 1);
现在来分析一下类型,self的类型应该是Func<Tself,Func<int,int>>,self(self)的返回类型正好是Func<int,int>,同时fact_maker也是Func<Tself,Func<int,int>>,可以顺利地被self传递,一切都很和谐。那Tself该怎么表达呢?它接受自己,且返回Func<int,int>,如此便得到了如下委托定义(Ouroboros是衔尾蛇的意思):
//C#
delegate T OuroborosFunc<T>(OroborosFunc<T> self);
这样OuroborosFunc<Func<int,int>>正是我们需要的self类型。到这里,可以测试一下以上的构想,看看是否能良好工作了。
//C#
OuroborosFunc<Func<int, int>> fact_maker =
self => x => x == 0 ? 1 : x * self(self)(x - 1);
int result = fact_maker(fact_maker)(5);
Console.WriteLine(result); //120
非常棒,输出了正确的结果。如果你愿意,可以不需要这个fact_maker,而是直接把这个Lambda应用到另一个同样的Lambda身上,不过需要注意显式声明类型,因为C#不允许Lambda自动类型推断,就像这样:
//C#
int result = ((OuroborosFunc<Func<int, int>>)
(self => x => x == 0 ? 1 : x * self(self)(x - 1)))
(self => x => x == 0 ? 1 : x * self(self)(x - 1))(5);
Console.WriteLine(result); //120
嗯,也不错,虽然丑陋了一点,但毕竟还是满足了之前对匿名递归的定义。当然,其中的自我调用部分可以简化成这样:
Func<int, int> Fact =
((Func<OuroborosFunc<Func<int, int>>, Func<int, int>>)(s => s(s)))
(self => x => x == 0 ? 1 : x * self(self)(x - 1));
问:可是,讲了半天,还是没有讲到Y组合子啊。
答:别急,我们姑且用一个例子来展示匿名递归,不具备足够的抽象性,离Y组合子还有不少的路要走,请耐性待续…
大到可以小说的Y组合子(一)的更多相关文章
- 大到可以小说的Y组合子(三)
答:关于Fix的问题你fix了吗? 问:慢着,让我想想,上次留下个什么问题来着?是说我们有了一个求不动点的函数Fix,但Fix却是显式递归的,是吧? 答:有劳你还记的这个问题. 问:Fix的参与背离了 ...
- 大到可以小说的Y组合子(二)
问:上一回,你在最后曾提到"抽象性不足",这话怎么说? 答:试想,如果现在需要实现一个其它的递归(比如:Fibonacci),就必须把之前的模式从头套一遍,然后通过fib_make ...
- 大到可以小说的Y组合子(零)
问:啊!我想要一个匿名的递归… 答:Y(音同Why)… … … 问:作为一位命令式语言的使用者,为什么会突然折腾起Y组合子呢? 答:的确,这事儿要从很久以前的几次搁浅开始说起…上学的时候,从来没有接触 ...
- Y组合子
Y组合子 Y组合子的用处 作者:王霄池链接:https://www.zhihu.com/question/21099081/answer/18830200来源:知乎著作权归作者所有.商业转载请联系作者 ...
- Lambda演算 - 简述Y组合子的作用
Y组合子:\f.(\x.f(xx))(\x.f(xx)),接受一个函数,返回一个高阶函数 Y组合子用于生成匿名递归函数. 什么叫匿名递归函数,考虑以下C语言递归函数 int sum(int n) { ...
- Racket中使用Y组合子
关于Y组合子,网上已经介绍很多了,其作用主要是解决匿名lambda的递归调用自己. 首先我们来看直观的递归lambda定义, 假设要定义阶乘的lambda表达,C#中需要这么定义 Func<in ...
- 简单易懂的程序语言入门小册子(4):基于文本替换的解释器,递归,如何构造递归函数,Y组合子
递归.哦,递归. 递归在计算机科学中的重要性不言而喻. 递归就像女人,即令人烦恼,又无法抛弃. 先上个例子,这个例子里的函数double输入一个非负整数$n$,输出$2n$. \[ {double} ...
- [学习] 从 函数式编程 到 lambda演算 到 函数的本质 到 组合子逻辑
函数式编程 阮一峰 <函数式编程初探>,阮一峰是<黑客与画家>的译者. wiki <函数编程语言> 一本好书,<计算机程序的构造与解释>有讲到schem ...
- 面向组合子设计Coder
面向组合子 面向组合子(Combanitor-Oriented),是最近帮我打开新世界大门的一种pattern.缘起haskell,又见monad与ParseC,终于ajoo前辈的几篇文章. 自去年9 ...
随机推荐
- iOS8中添加的extensions总结(二)——分享扩展
分享扩展 注:此教程来源于http://www.raywenderlich.com的<iOS8 by Tutorials> 1.准备 这次例子来源于国外的图片分享网站Imgur.com 首 ...
- (六)Angularjs - 启动引导
自动引导 AngularJs 通过 ng-app 指令进行自动引导 手工引导启动框架 如果一个HTML文件中 有多个ng-app,AngularJS只会自动引导启动它找到的第一个ng-app应用,这是 ...
- C++拾遗(六)函数相关(1)
返回值 C++规定返回值不能是 数组.但可以是其它任何类型(包括结构体和对象). 通常,函数将返回值复制到指定的CPU寄存器或内存单元中,然后调用函数调用该内存单元的值. 函数原型 参数列表中可以不包 ...
- basename usage in linux
作用:去掉文件的目录和后缀 1.去掉文件路径 jenkins@work:~/ci/script$ basename /backup/jenkins/ci/script/Release.sh.bak R ...
- ios nslog 打印字典为中文
#import <Foundation/Foundation.h> @implementation NSDictionary (Log) - (NSString *)description ...
- mysql 获取当前日期及格式化
MYSQL 获取当前日期及日期格式获取系统日期: NOW()格式化日期: DATE_FORMAT(date, format)注: date:时间字段format:日期格式 返回系统日期,输出 2009 ...
- UCOS 信号量的初值问题
当 pend请求发出的时候信号量的值减1,当post的时候信号量的值加1,信号量的值0跟1分别是用来同步跟互斥的,什么是同步,什么是互斥呢...假设你把信号量的值设为0,有A,B连个任务,当A发出pe ...
- Andriod手势密码破解
★ 引子 之前在Freebuf上看到一片文章讲Andriod的手势密码加密原理,觉得比较有意思,所以就写了一个小程序试试. ★ 原理 Android的手势密码加密原理很简单: 先 ...
- ASP.NET MVC5中的数据注解
ASP.NET MVC5中Model层开发,使用的数据注解有三个作用: 数据映射(把Model层的类用EntityFramework映射成对应的表) 数据验证(在服务器端和客户端验证数据的有效性) 数 ...
- HashMap HashTable HashSet区别剖析
HashMap.HashSet.HashTable之间的区别是Java程序员的一个常见面试题目,在此仅以此博客记录,并深入源代码进行分析: 在分析之前,先将其区别列于下面 1:HashSet底层采用的 ...