6

我正在运行 Chez Scheme 9.5 并尝试在库中定义语法转换器。这是一个例子:

(library (forlib)
  (export for)
  (import (rnrs (6)))

  (define-syntax for
    (syntax-rules (in)
      [(for x in lst body1 body2 ...)
       (for-each (lambda (x) body1 body2 ...) lst)])))

我将其保存在一个文件中forlib.sschez从同一目录运行。然后在 REPL 中,我得到了这个:

> (import (forlib))
> (for x in '(1 2 3) (display x))
Exception: invalid syntax (for x in (quote (1 2 3)) (display x))
Type (debug) to enter the debugger.

如果我将语法定义更改为

(define-syntax for
  (syntax-rules ()
    [(for x lst body1 body2 ...)
     (for-each (lambda (x) body1 body2 ...) lst)])))

(没有in关键字),一切正常:

> (import (forlib))
> (for x '(1 2 3) (display x))
123
> _

回到带有in关键字的旧定义。如果我将测试代码放入测试文件中:

;;; test-for.ss
(import (rnrs (6))
        (forlib))

(for x in '(1 2 3) (display x))

并尝试执行该文件,结果取决于我如何执行该文件。如果我使用 运行这个程序chez --program,它会按预期工作:

$ chez --program test-for.ss
123
$ _

如果我使用 运行它chez --script,我会得到与上面相同的错误:

$ chez --script test-for.ss
Exception: invalid syntax (for x in (quote (1 2 3)) (display x)) at line 6, char 1 of test-for.ss
$ _

这提出了两个问题:

  • 为什么 REPL--script可以很好地导入没有特殊关键字的语法形式,但拒绝接受其中包含特殊关键字的语法形式?
  • --script和 和到底有什么区别--program?用户手册说这--program意味着文件内容被解释为 rnrs 顶级程序,但对语义是什么保持沉默 --script

最后,为了彻底解决我的困惑,如果我直接在 REPL 中输入上述语法定义,那么一切都按预期工作:

> (define-syntax for
    (syntax-rules (in)
      [(for x in lst body1 body2 ...)
       (for-each (lambda (x) body1 body2 ...) lst)])))
> (for x in '(1 2 3) (display x))
123
> _

那么从库中导入的语法转换器和直接在 REPL 中定义的语法转换器在 REPL 中有何不同?

4

1 回答 1

5

在 Chez Scheme 中,操作模式之间的一些语义不同。我将把这些模式命名为 r6rs 和 repl。

R6RS:

  • 程序(命令行上的--program)
  • 图书馆

回复:

  • 脚本(命令行上的--script)
  • 文件(使用加载过程或作为不带 --program 或 --script 的 CLI 参数)
  • 当然还有repl

对于您的特定问题,如果您希望 repl 和脚本匹配库和程序的行为,则需要定义和导出一个名为infrom的标识符。forlib这可以像空定义一样简单,或者您可以创建一个在宏体外部使用时引发错误的语法定义。简单的:

(library (forlib)
  (export for in)
  (import (rnrs))

  (define in)

  (define-syntax for ...)
)

句法定义:

(library (forlib)
  (export for in)
  (import (rnrs))

  (define-syntax in
    (identifier-syntax
      (error #f "Misplaced aux syntax in")))

  (define-syntax for ...)
)

两者都工作正常;使用任何你喜欢的。

repl 和 r6rs 模式之间的其他差异——据我所知——源于它们如何评估表达式。在 repl 模式下,每个表达式都单独处理。也就是说,Chez 读取一个表达式,评估该表达式,并可能打印结果。在 r6rs 模式下,文件的全部内容作为一个单元进行评估。明确地说,repl 模式的顶级延续是“读取、评估、可能打印和循环”,而 r6rs 模式的顶级延续是下一个表达式。例如,此代码在作为程序或脚本运行时会表现不同:

(import (chezscheme))

(define println)

(printf "~x\n"
  (call/cc (lambda (k)
             (set! println k)
             1)))

(println 5)
(println 6)

另一个区别是,在 r6rs 模式下,您不能定义一个函数,然后在语法引用之外的语法案例宏扩展中使用它。

(define ($two) 2)

(define-syntax two
  (lambda (x)
    (syntax-case x ()
      [_ ($two)])))
于 2019-01-15T00:15:42.283 回答