函数式编程
Lambda 演算 #01
在定义了布尔值并且实现了“布尔运算”之后,我们要迈入下一步探索:自然数在 Lambda 演算中应该如何定义?
自然数的定义
我们依然采用 Alonzo Church 的方案来构造所谓的邱奇数(Church numerals),这个方案也满足皮亚诺公理,可以形成一个皮亚诺算术系统。
介绍这一方案前,我想定义一下函数的复合的表示形式,以便后文使用:
f∘n=n timesf∘f∘⋯∘f
而这一表示方案即利用函数复合的次数来表示对应的自然数,同时0以不应用函数来表示,写作定义的形式即:
012N:=λ f x.x:=λ f x.f x:=λ f x.f (f x)…:=λ f x.f∘n x
为避免歧义,我们定义的自然数用N表示,而其对应的数字意义的N用小写字母n表示。
可以发现,我们的0和F是α等价的,这对于学过编程——尤其是 C 语言——的同学似乎是理所当然的:
毕竟计算机里确实就是用 0 去表示 False。
后继算子的构造
根据皮亚诺公理,现在需要得到一个后继算子succ来实现自然数的递增,即我们希望有succ 0=1,succ 1=2,succ 2=3以此类推。将其展开,即:
succ (λ f x.x)succ (λ f x.f x)succ (λ f x.f (f x))=λ f x.f x=λ f x.f (f x)=λ f x.f (f (f x))…
这意味着succ算子将接收一个“自然数 n”作为参数并且返回“自然数 n+1”,但注意,这里的“自然数”的本质亦是一个函数。
由上篇文章关于柯里化的讨论,我们可以认为succ算子应该是一个“三元函数”,它的第一个参数是一个“自然数”,而第二、第三个参数则是“自然数”本身应该具有的两个参数,如此,当我们给定一个“自然数 n”时,它的值便成为了我们所需要的“自然数”的样子。
在这样的思路下,succ的作用就很显而易见了:
succ N f x=f (N f x)
于是,很显然得:
succ:=λ N f x.f (N f x)
后继之上——加法,乘法与指数
有了后继算子,我们可以思考如何构造加法和乘法算子了。
由于我们定义了“自然数”的 Lambda 项是函数的复合,那么对于函数复合的一些基本的运算律我们就可以拿来使用并借此进行算子的构造。
首先,是加法算子,函数复合n+m次的意义其实可以理解为:函数复合n次,而后复合m次。即:
f∘(n+m)(x)=f∘n(f∘m(x))
于是,利用构造后继算子的经验,加法算子plus理应是一个“四元函数”,且返回值是两个“自然数”的“复合”:
plus:=λ N M f x.M f (N f x)
其次,轮到了乘法算子,函数复合n×m次的意义其实可以理解为:函数复合n次后作为一个新的函数,而后让这个新的函数复合m次。即:
f∘(n×m)(x)=(f∘n)∘m(x)
和加法算子极其类似,不过在乘法算子中我们需要注意的是f需要变为(N f),而不变的是x:
mult:=λ N M f x.M (N f) x
对于指数算子的构造,我们需要先从指数到底做了什么这件事情入手。一个函数复合nm次可以理解为:函数复合n次后作为一个新的函数继续复合n次,将这个过程持续进行m遍。即:
f∘(nm)(x)=m times((f∘n)∘n…)∘n(x)
这会可能有点不好办了,但别急,我们看一下“自然数”的定义:(N f)可以做到将f复合n次得到f∘n,而“自然数”N又可以理解为一个复合函数的函数——但它依旧是个函数——虽然它会接受两个参数。
于是,我们产生了一个想法:是否可以将N复合呢?如果N→M,f→N是不是就能实现将N复合M次了呢?我们尝试着计算一下如下的 Lambda 项:
M N f x=N (N…N (N (N_m times f)))x=N (N…N (N_m−1 times (f∘n )))x=N (N…N_m−2 times (f∘(n2 ) ))x…=N (N…_m−k times (f∘(nk ) ))x=(f∘(nm ) ) x
而这也就是我们所期望得到的指数算子了,它看上去十分简单:
exp:=λ N M.M N
前驱和减法
为了定义减法算子,我们需要先定义一个前驱算子。而定义前驱算子是个比较困难的过程,为了实现它有两个方法,这里我选择较为简单的一条,另一条你可以在Church encoding - Wikipedia中找到较为详细的描述。
首先,我们先定义一个数对算子pair,将两个数组成一对来用:
pair:=λ M N x.x M N
为了方便,我们将他记作:[a,b]:=pair a b=λx.x a b,利用之前定义的布尔值的选择能力,我们可以用它来选择其中的某一项:
[a,b] T[a,b] F=pair a b T=T a b=a=pair a b F=F a b=b
然后,我们再为pair定义一个迭代算子phi,它的作用是将数对[n,m]忽略m的值,并迭代到[n+1,n],即在自增的同时能将自己的前驱保存在后一位:
phi:=λ p.[succ (p T),p T]
比如:
phi [0,0]phi [5,4]=[succ 0,0]=[1,0]=[succ 5,4]=[6,5]
于是,我们可以借助”自然数“的复合迭代能力让数对在”加上来“的同时给了我们得到其前驱的机会,即定义pred算子的机会:
pred:=λN.N phi [0,0] F
其意义就是让pair[0,0]进行N次phi而后选取其后一位输出。
用 3 作为例子试一下:
pred 3=3 phi [0,0] F=2 phi [1,0] F=phi [2,1] F=[3,2] F=2
而对于”自然数“0,依照皮亚诺公理,它没有前趋,而这里则用稍微变通一点的方式,不妨设定 0 的前趋就是 0 本身。在这种情况下,我们的pred算子依然可用,检验一下:
pred 0=0 phi [0,0] F=[1,0] F=0
完美符合预期。而有了前驱算子pred之后,定义减法算子minus就显得很简单了:如果要计算n−m则只需要让n连续求m次前驱即可:
minus:=λ N M.M pred N
而注意到我们定义的运算是在自然数的范围内讨论,所以对于算子minus,当m>n时,minus N M=0
而对于仅剩的除法运算,由于它较为复杂,暂时先不作讨论,你可以在Division找到相关内容。
关系运算
在定义完基础的算数运算和布尔运算之后,我们的脚步可以迈入关系运算的领域了。
首先,我们考虑应该算子:IsZero,当输入为0输出为T,当输入为其他自然数时,输出为F。
IsZero:=λ N.N F not F
这里需要用到上篇文章中关于F的一个特殊用处:F x=λ x.x=I
这个算子的核心在于(N F not)。当N为0时,相当于F F not=not。
IsZero 0=not F=T
当N是大于 0 的自然数时,相当于:
IsZero N=N F not F=F(F…F(F_n>0 times not)) F=F(F…F_n−1 times I)) F=I F=F
于是我们可以更加轻松得使用之前的准备去构建相等算子EQ,小于或等于算子LEQ,大于算子GT,不等算子NEQ等等了。利用一些很简单的运算律:
m≤nm=n⇔m−n=0⇔(m≤n)∧(n≤m)
我们可以得到:
LEQEQGT=λ M N.IsZero (minus M N)=λ M N.and (LEQ N M) (LEQ N M)=λ M N.not (LEQ N M)
不妨试试构造更多的关系算子吧!
未完待续… 下一篇:#02 | Lambda 演算中的循环与递归
Reference
- Functional programming - Wikipedia
- Lambda calculus - Wikipedia
- λ演算 - 函数式语言的起源 - 知乎
- 神奇的λ演算 - 简书
- Church encoding - Wikipedia