如何用 MoonBit 实现无类型 Lambda 演算解释器?

2023-11-30 10:31:44 +08:00
 Moonbit

相信点开这篇文章的你或多或少地听说过函数式编程这个名词。在摩尔定律失效的今天,对多核处理器的充分利用成为了一种越发重要的程序优化方法,而函数式编程也因为其并行运算亲和的特点在大众视野中越发频繁地出现。究其原因,离不开它从其理论上的祖先之一—Lambda 演算那里所继承的特征。

而 Lambda 演算这一起源于 20 世纪 30 年代,出自图灵导师阿隆佐·邱奇之手的形式系统如今已经发展成了蔚为大观的一个大家族,本文将展示其中最基础的一种:无类型 Lambda 演算(这也是最早阿隆佐·邱奇提出的那种)。

无类型 Lambda 演算的基本规则

无类型 Lambda 演算中能做的事情只有定义 Lambda (经常称为 Abstraction )和调用Lambda (经常称为 Application ),它们也是 Lambda 演算中的基础表达式。

由于函数式编程范式对主流编程语言的影响,大多数程序员对 Lambda 表达式这个名字已经不会感到陌生了。不过,无类型 Lambda 演算中的 Lambda 要比主流编程语言简单一些。一个 Lambda 通常看起来就像这样:λx.x x , 其中x是它的参数(每个 Lambda 只能有一个参数),.是分隔参数与表达式具体定义的分隔符,后面的 x x 便是它的定义了。

也有些材料的记法不写空格,上面的例子要改写成λx.xx

上面的x x如果换成x(x),可能更符合我们在一般语言中见到的函数调用。但在 Lambda 演算较常见的写法中,调用一个 Lambda 只需要在它和它的参数中间写个空格。此处我们调用 x 所给出的参数就是 x 自己。

以上两种表达式和定义 Lambda 时引入的变量加在一起合称 Lambda 项,我们在 MoonBit 里用一个 enum 类型来表示它。

enum Term {
  Var(String) // 变量
  Abs(String, Term) // 定义 lambda ,变量用字符串表示
  App(Term, Term) // 调用 lambda
}

我们在日常编程中所接触的概念诸如布尔值、if 表达式、自然数算术乃至递归都可以通过 Lambda 表达式实现,但这并非本文内容的重心所在,有兴趣的读者可以参阅:

要实现一个无类型 Lambda 演算的解释器,我们所需要了解的基本就只有两条规则:Alpha 转换与 Beta 归约。

Alpha 转换所描述的事实是,Lambda 的结构是重点,变量名叫什么没那么重要。λx.xλfoo.foo可以互相替换。对于某些有重复变量名的嵌套 Lambda 例如λx.(λx.x) x,重命名时不能把内层的变量也重命名了,例如上面的例子可以通过 Alpha 转换写成λy.(λx.x) y

Beta 归约则专注于处理 Lambda 的调用,还是举一个例子:

(λx.(λy.x)) (λs.(λz.z))

在无类型 Lambda 演算中,调用 Lambda 之后所需要做的事情仅仅是对参数进行替换(substitution),上面这个例子里就需要把变量 x 替换成(λs.(λz.z)),得到的结果是:

(λy.(λs.(λz.z)))

想看更多的例子可以参见:程序语言理论与实现

自由变量与变量捕获

一个 Lambda 项中的变量如果在它所处的上下文中没有定义,那么我们叫它自由变量。例如(λx.(λy.fgv h))这个 Lambda 项中变量fgvh就没有对应的 Lambda 定义。

在进行 Beta 归约时,如果用于替换变量的那个 Lambda 项中含有自由变量,可能会导致一种被称为“变量捕获“的行为:

(λx.(λy.x)) (λz.y)

上面这个表达式在替换后会变成:

λy.λz.y

λz.y中的自由变量被当成了某个 Lambda 的参数,这显然不是我们想要的。

变量捕获问题在编写解释器时的常见解决方案是在替换前遍历表达式得到一个自由变量的集合,做替换时遇到内层 Lambda 就判断一下变量名在不在这个自由变量集合里面:

// (λx.E) T => E.subst(x, T)
fn subst(self : Term, var : String, term : Term) -> Term {
  let freeVars : Set[String] = term.get_free_vars()
  match self {
    Abs(variable, body) => {
      if freeVars.contains(variable) {
        //自由变量集合中有当前这个内层 lambda 的参数名,即会发生变量捕获
        abort("subst(): error while encountering \(variable)")
      } else {
        ......
      }
    }
    ......
  }
}

接下来我们介绍一种较少见但具有一定便利性的方法:de Bruijn index (德布朗系数)。

de Bruijin index (德布朗系数)

de Bruijn index (德布朗指数)是一种用整数表示 Lambda 项中变量的技术,具体地说,它用变量所在位置和原先引入它的位置中间有几层 Lambda 来替换特定变量。

λx.(λy.x (λz.z z))

λ.(λ.1 (λ.0 0))

上面的例子中,变量 x 和引入它的地方λx中间有一个λy,于是将x替换为1,而z和定义它的位置中间没有夹杂其他的 Lambda ,于是直接用0替换。某种程度上说,德布朗指数的值描述的是变量与对应 Lambda 的相对距离,此处的距离衡量标注就是中间嵌套的 Lambda 层数。

同一个变量在不同的地方可能会用不同的整数来替换

我们定义一个新类型TermDBI来表示使用德布朗指数的 Lambda 项:

enum TermDBI {
  Var(String, Int)
  Abs(String, TermDBI)
  App(TermDBI, TermDBI)
}

不过直接编写以及阅读德布朗指数形式的 Lambda 很痛苦,所以我们需要编写一个将Term转换成TermDBI的函数bruijn() - 这也是TermDBI类型定义中仍有String的原因,保留原变量名可用于它的to_string方法,这样就可以方便地用println打印求值结果查看了。

fn to_string(self : TermDBI) -> String {
  match self {
    Var(name, _) => name
    Abs(name, body) => "(\\\(name).\(body))"
    App(t1, t2) => "\(t1) \(t2)"
  }
}

为了简化实现,如果输入的Term中含有自由变量,bruijn()函数会直接报错。MoonBit 将会在标准库中提供Result[V, E]类型,它有Ok(V)Err(E)两个构造器,分别代表计算成功与失败。不过,目前我们还是得自己定义一下:

enum Result[V, E] {
  Ok(V)
  Err(E)
}

使用过 Rust 语言的读者应该会感到熟悉

fn bruijn(self : Term) -> Result[TermDBI, String]

我们采取一种笨办法来保存变量名与相关联的嵌套深度,首先定义Index类型:

struct Index {
  name : String
  depth : Int
}

然后写一个从List[Index]中根据特定name查找对应depth的辅助函数:

// 查找环境中第一个 varname 对应的整数
fn find(map : List[Index], varname : String) -> Result[Int, String] {
  match map {
    Nil => Err(varname) // abort("variable \'\(varname)\' not found")
    Cons(i, rest) => {
      if i.name == varname {
        Ok(i.depth)
      } else {
        find(rest, varname)
      }
    }
  }
}

现在可以补全bruijn()函数了。

fn go(m : List[Index], t : Term) -> Result[TermDBI, String] {
    match t {
      Var(name) => {
        let idx = find(m, name)
        match idx {
          Err(name) => Err(name)
          Ok(idx) => Ok(Var(name, idx))
        }
      }
      Abs(varname, body) => {
        let m = m.map(fn (index){
          { name : index.name, depth : index.depth + 1 }
        }).cons({ name : varname, depth : 0 })
        let res = go(m, body)
        match res {
          Err(name) => Err(name)
          Ok(term) => Ok(Abs(varname, term))
        }
      }
      App(e1, e2) => {
        match (go(m, e1), go(m, e2)) {
          (Ok(e1), Ok(e2)) => Ok(App(e1, e2))
          (Err(name), _) => Err(name)
          (_, Err(name)) => Err(name)
        }
      }
    }
  }
  go(Nil, self)

TermDBI上做归约

归约主要处理的是App,即调用:

fn eval(self : TermDBI) -> TermDBI {
  match self {
    App(t1, t2) => {
      match (eval(t1), eval(t2)) {
        (Abs(_, t1), t2) => eval(subst(t1, t2))
        (t1, t2) => App(t1, t2)
      }
    }
    Abs(_) => self
    otherwise => abort("eval(): \(otherwise) ")
    // eval 应该不会遇到自由变量才对
  }
}

首先对两个子项尝试归约,然后看eval(t1)得到的是否是一个 Lambda ,如果是,就执行一步变量替换(通过 subst 函数)然后继续化简。对于 Lambda(即Abs),直接原样返回即可。

subst 函数的实现在不用考虑自由变量的情况下简单了许多,只要记录递归到当前位置的深度并且与遇到的变量进行比对,大小相等就是需要替换的目标变量。

fn subst(t1 : TermDBI, t2 : TermDBI) -> TermDBI {
  fn go(t1 : TermDBI, t2 : TermDBI, depth : Int) -> TermDBI {
    match t1 {
      Var(name, d) => {
        if d == depth {
          t2
        } else {
          t1
        }
      }
      Abs(name, t) => {
        Abs(name, go(t, t2, depth + 1))
      }
      App(tl, tr) => {
        App(go(tl, t2, depth), go(tr, t2, depth))
      }
    }
  }
  go(t1, t2, 0)
}

完整代码实现

改进

我们在保存变量名到索引的对应关系时使用了List[Index]类型,并且在每增加一层 Lambda 时更新整个列表,但是这其实是个很笨的办法。相信聪明且注意力集中的你们很快就能发现,其实只需要保存一个List[String]就够了,大家如果有兴趣也可以自己试着做一做。

701 次点击
所在节点    编程
0 条回复

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/996447

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX