这个系列有个显著的特点,那就是标题越来越长。忽然发现今天是读书节,读书节多读书。

==下面是没有意义的一段话================================================

我是一个喜欢从学习知识中获得乐趣并乐于分享这种乐趣的人。我认为大部分知识只要花点时间都是能学会的。几年前,我迷上微分几何。我对每个朋友说这东西很有意思花点时间精力就能学会。他们回答说唉没时间时间不知去哪儿了。后来,我迷上量子力学。我对每个朋友说这东西值得一学,只要花点时间精力。他们回答说唉烦心事太多没精力去学。再后来,我迷上程序语言。我对每个朋友说只要使劲挤点时间就能掌握这个。他们回答说唉浑浑噩噩不知道一直在忙什么没时间唉。现在,他们都结婚了,我依然单身。

==然后,正文开始======================================================

不动点

唉,又是一个数学概念, 又是没有实际意义的一节。 但是作为递归相关的常识,还是得提一下。 话说回来,把这玩意儿当作常识,我大概是价值观扭曲,重要性排行颠倒吧。 据说很多研究程序语言,研究偏微分方程的人都这样。

然后,一个——呃——东西$x$被称为函数$f$的不动点,如果它满足下面条件: \[ (f \; x) = x \]

不动点和递归有什么关系呢? $({Y} \; f)$不仅是关于辅助函数$f$的递归函数,同时也是$f$的不动点。 证明如下: \begin{eqnarray*} ({Y} \; f) &=& (\lambda x.(\lambda v.(v \; v) \; \lambda f.(x \; (f \; f))) \; f) \\               &=& (\lambda x.(\lambda v.(v \; v) \; \lambda p.(x \; (p \; p))) \; f) \\               &=& (\lambda v.(v \; v) \; \lambda p.(f \; (p \; p))) \\               &=& (\lambda p.(f \; (p \; p)) \; \lambda p.(f \; (p \; p))) \\               &=& (f \; (\lambda p.(f \; (p \; p)) \; \lambda p.(f \; (p \; p)))) \\               &=& (f \; (\lambda v.(v \; v) \; \lambda p.(f \; (p \; p)))) \\               &=& (f \; (\lambda x.(\lambda v.(v \; v) \; \lambda p.(x \; (p \; p))) \; f)) \\               &=& (f \; ({Y} \; f)) \end{eqnarray*} 由于递归函数肯定能写成$({Y} \; f)$的形式(参见之前mkdouble的构造方法),递归函数必定是某个函数$f$的不动点。

加入fix表达式

用Y组合子构造递归函数总归太麻烦。 所以需要能直接构造递归函数的表达式。 先来看这个表达式需要哪些元素。 首先,递归函数要调用自己,需要一个指向自己的变量,记为$X_1$。 其次,递归函数有个参数,记为$X_2$。 最后,递归函数的函数体,记为$M$。 构造递归函数其实是寻找不动点的过程,所以这个表达式叫fix表达式(不动点叫fixed point)。 fix表达式长这个样: \[ ({fix} \; X_1 \; X_2 \; M) \] 加入fix表达式最简单的方法是定义为宏: \[ ({fix} \; X_1 \; X_2 \; M) = ({Y} \; \lambda X_1.\lambda X_2.M) \] 但是这样做又重新引入了Y组合子。 这次采用另一种方法,不定义宏,而是把fix表达式加入语法: \begin{eqnarray*}   M, N, L &=& ... \\           &|& ({fix} \; X_1 \; X_2 \; M) \end{eqnarray*}

现在考察fix表达式的求值过程。 fix表达式的求值结果是个函数,参数是$X_2$: \[ eval(({fix} \; X_1 \; X_2 \; M)) = \lambda X_2.? \] 最适合放在问号处的函数体的是$M$,但是$M$包含了自由变量$X_1$,使用$M$之前得先想方法除掉$X_1$。 $X_1$代表递归函数本身,也就是$({fix} \; X_1 \; X_2 \; M)$。 为了去掉$M$中的自由变量$X_1$,将$X_1$替换为递归函数$({fix} \; X_1 \; X_2 \; M)$。 综上,fix表达式的求值过程为: \[ eval(({fix} \; X_1 \; X_2 \; M)) = \lambda X_2.M[X_1 \leftarrow ({fix} \; X_1 \; X_2 \; M)] \] 代码:

加入了新语法,要添加相应的替换过程(这个替换过程和$\lambda X.M$的替换过程类似,但是麻烦了点): \begin{eqnarray*}   ({fix} \; X_1 \; X_2 \; M)[X_1 \leftarrow N] &=& ({fix} \; X_1 \; X_2 \; M) \\   ({fix} \; X_1 \; X_2 \; M)[X_2 \leftarrow N] &=& ({fix} \; X_1 \; X_2 \; M) \\   ({fix} \; X_1 \; X_2 \; M)[X_3 \leftarrow N] &=& ({fix} \; X_4 \; X_5 \; M[X_2 \leftarrow X_5][X_1 \leftarrow X_4][X_3 \leftarrow N]) \\   &其中&X_3 \neq X_1, X_3 \neq X_2, \\   &&X_4 \notin FV(N), X_4 \notin FV(M)\backslash\{X_1\}, \\   &&X_5 \notin FV(N), X_5 \notin FV(M)\backslash\{X_2\} \end{eqnarray*} 代码:

测试一下:

'((fix f n (if (iszero n) 0 (+ 2 (f (- n 1))))) 4)

>> 8

加入letrec表达式

let表达式不能定义递归函数,所以有个letrec表达式专门用来定义递归函数。 letrec表达式长这个样: \[ ({letrec} \; X_1 \; X_2 \; N \; M) \] $X_1$,$X_2$和$N$定义了一个递归函数,$M$是用到这个递归函数的一个表达式。

类似let表达式,将letrec表达式定义为宏: \[ ({letrec} \; X_1 \; X_2 \; N \; M) = (\lambda X_1.M \; ({fix} \; X_1 \; X_2 \; N)) \] 代码:

简单易懂的程序语言入门小册子(5):基于文本替换的解释器,递归,不动点,fix表达式,letrec表达式的更多相关文章

  1. 简单易懂的程序语言入门小册子(3):基于文本替换的解释器,let表达式,布尔类型,if表达式

    let表达式 let表达式用来声明一个变量. 比如我们正在写一个模拟掷骰子游戏的程序. 一个骰子有6个面. 所以这个程序多次用到了6这个数字. 有一天,我们忽然改变主意,要玩12个面的骰子. 于是我们 ...

  2. 简单易懂的程序语言入门小册子(1):基于文本替换的解释器,lambda演算

    最近比较闲,打算整理一下之前学习的关于程序语言的知识.主要的内容其实就是一边设计程序语言一边写解释器实现它.这些知识基本上来自Programming Languages and Lambda Calc ...

  3. 简单易懂的程序语言入门小册子(1.5):基于文本替换的解释器,递归定义与lambda演算的一些额外说明

    这一篇接在第一篇lambda演算的后面.讲讲一些数学知识. 经常有些看似很容易理解的东西,一旦要描述得准确无误,就会变得极为麻烦. 软件工程里也有类似情况:20%的代码实现了核心功能,剩下80%的代码 ...

  4. 简单易懂的程序语言入门小册子(6):基于文本替换的解释器,引入continuation

    当我写到这里的时候,我自己都吃了一惊. 环境.存储这些比较让人耳熟的还没讲到,continuation先出来了. 维基百科里对continuation的翻译是“延续性”. 这翻译看着总有些违和感而且那 ...

  5. 简单易懂的程序语言入门小册子(7):基于文本替换的解释器,加入continuation,重构解释器

    或许在加入continuation之前要先讲讲费这么大劲做这个有什么意义. 毕竟用不用continuation的计算结果都是一样的. 不过,这是一个兴趣使然的系列,学习这些知识应该完全出于好奇与好玩的 ...

  6. 简单易懂的程序语言入门小册子(4):基于文本替换的解释器,递归,如何构造递归函数,Y组合子

    递归.哦,递归. 递归在计算机科学中的重要性不言而喻. 递归就像女人,即令人烦恼,又无法抛弃. 先上个例子,这个例子里的函数double输入一个非负整数$n$,输出$2n$. \[ {double} ...

  7. Go语言入门篇-gRPC基于golang & java简单实现

    一.什么是RPC 1.简介: RPC:Remote Procedure Call,远程过程调用.简单来说就是两个进程之间的数据交互. 正常服务端的接口服务是提供给用户端(在Web开发中就是浏览器)或者 ...

  8. C语言入门(2)——安装VS2013开发环境并编写第一个C语言程序

    在C语言入门系列中,我们使用Visual studio 2013 Professional作为开发工具.本篇详细介绍如何安装Visualstudio 2013 Professional并写出我们第一个 ...

  9. 《Java从入门到失业》第一章:计算机基础知识(三):程序语言简介

    1.3程序语言简介 我们经常会听到一些名词:低级语言.高级语言.编译型.解释型.面向过程.面向对象等.这些到底是啥意思呢?在正式进入Java世界前,笔者也尝试简单的聊一聊这块东西. 1.3.1低级语言 ...

随机推荐

  1. https跨域到http问题解决

    整体结构 用户--https-->Nginx--http-->Tomcat 页面中包含了iframe,iframe的src也是Https的,但是当frame里面的内容表单提交后tomcat ...

  2. spring-boot (四) springboot+mybatis多数据源最简解决方案

    学习文章来自:http://www.ityouknow.com/spring-boot.html 配置文件 pom包就不贴了比较简单该依赖的就依赖,主要是数据库这边的配置: mybatis.confi ...

  3. Java设计模式之《构建者模式》及应用场景

    原创作品,可以转载,但是请标注出处地址:http://www.cnblogs.com/V1haoge/p/6553374.html 构建者模式,又称建造者模式,将一部负责对象的构建分为许多小对象的构建 ...

  4. #19 re&jieba模块

    前言 在Python中,需要对字符串进行大量的操作,有时需要从一个字符串中提取到特定的信息,用切片肯定是不行的,所有这一节记录两个强大的文本处理模块,一个是正则表达式re模块,另一个是中文处理模块ji ...

  5. Python系列:一、Python概述与环境安装--技术流ken

    Python简介 Python是一种计算机程序设计语言.是一种动态的.面向对象的脚本语言,最初被设计用于编写自动化脚本(shell),随着版本的不断更新和语言新功能的添加,越来越多被用于独立的.大型项 ...

  6. MyBatis学习总结(二)——MyBatis核心配置文件与输入输出映射

    在上一章中我们学习了<MyBatis学习总结(一)——ORM概要与MyBatis快速起步>,这一章主要是介绍MyBatis核心配置文件.使用接口+XML实现完整数据访问.输入参数映射与输出 ...

  7. [nodejs] nodejs开发个人博客(三)载入页面

    模板引擎 使用ejs作为我们博客的前端模板引擎,用来从json数据生成html字符串 安装:npm install ejs -save 使用:入口文件中写入下面代码,定义/view/目录为视图目录 / ...

  8. git获取远程服务器的指定分支

    昨天糗大了...进入新公司,公司服务器上有Online为线上版本,开发版本默认的为Master,本地clone的开发版为master,公司用的git 自动部署(puh后服务器自动更新了代码...这个有 ...

  9. 【Java每日一题】20170221

    20170220问题解析请点击今日问题下方的“[Java每日一题]20170221”查看(问题解析在公众号首发,公众号ID:weknow619) package Feb2017; public cla ...

  10. MySQL事务(学习笔记)

    MySQL事务主要用于处理操作量大,复杂度高的数据.比如说,在人员管理系统中,你删除一个人员,你即需要人员的基本资料,也要删除和该人员相关的信息,如信箱,文章等等,这样,这些数据库操作语句就构成一个事 ...