Racket 中是否有返回过程的 lambda 表达式的函数?

Ash*_*orf 5 lisp lambda common-lisp racket

在 Common Lisp 的一个变体中(我认为是 CMUCL,但我可能错了——我再也找不到它了)有一个函数(我认为)被称为function-lambda-expression. 如果它有一个过程,它会打印出生成它的 lambda 表达式。例子:

(let ((my-thunk (lambda () (+ 1 2))))
    (write my-thunk)
    (write (function-lambda-expression my-thunk)))
Run Code Online (Sandbox Code Playgroud)

这将打印出如下内容:

#<PROCEDURE>
(LAMBDA () (+ 1 2))
Run Code Online (Sandbox Code Playgroud)

它对于调试和探索语言非常有用。

我正在 Racket 中寻找这样的功能。我已经浏览了Racket 文档,但找不到类似的内容。(但是,如果我忽略了它,我不会感到惊讶。)Racket 中是否有等价物?

Rya*_*per 8

lambda不可以。Racket生成的闭包不记得其 S 表达式(或语法对象)形式。它通常会记住它的名称(或者它的缩写源位置,如果不能推断出名称),这通常足以帮助调试。(见object-name。)

您可以lambda使用 Racket 的适用结构和简单的宏构建自己的具有此功能的变体。这是一个基本示例:

#lang racket

(struct exp-closure (f exp)
  #:property prop:procedure (struct-field-index f))

(define-syntax-rule (exp-lambda formals . body)
  (exp-closure (lambda formals . body)
               (quote (exp-lambda formals . body))))

(let ([my-thunk (exp-lambda () (+ 1 2))])
  (printf "fun is ~v\n" my-thunk)
  (printf "exp is ~v\n" (exp-closure-exp my-thunk))
  (printf "result is ~v\n" (my-thunk)))
Run Code Online (Sandbox Code Playgroud)

这产生

fun is #<procedure:...tmp/lambda.rkt:11:19>
exp is '(exp-lambda () (+ 1 2))
result is 3
Run Code Online (Sandbox Code Playgroud)

此宏的更好版本可能会将宏使用的源位置传播到lambda它创建的表达式或推断的名称(请参阅 参考资料syntax-local-infer-name),或两者兼而有之。