8 可爱的宏

有时候,一门语言的语法让你感觉不爽,如果能够做点微调,该多好啊?比如说Racket提供了 lambdaλ 来生成匿名函数,lambda对我来说太长,而λ在我的mbp的键盘上无法直接输入(不像ƒ这样的字符可以 ALT+f 输入,而是需要打开greek语言),如果我想用 ƒ 定义函数,该怎么做?

在Python中我还还真不知道该怎么做,也没这么想过,C可以用宏替换:

#define ƒ λ

所以在C里,很多时候宏的写法需要特殊处理,比如说:

#define MACRO(x, y) do { \
        // macro body		 \
while(0)
}
很丑很暴力。

不过我们知道,C的宏替换仅仅是预处理阶段基本不太考虑语法的情况下直接对目标做字符串替换,所以这么做有很多潜在的风险。那么Racket呢?能不能直接用 define 定义一下?

> (define ƒ λ)

eval:235:0: λ: bad syntax

  in: λ

好吧,语法错误。define 实际上是把后一个表达式求值然后绑定到 ƒ 变量上,而这里 λ 不是个可求值的表达式。都说lisp强大到可以生成任何编程语言,那么这么个小问题该怎么解决?简单:

> (define-syntax-rule (ƒ x y)
    (λ x y))
> (define area (ƒ (r) (* pi r r)))
> (area 10)

314.1592653589793

这就是我们这一章要讲的宏。