将字符串转换为lambda /值

时间:2015-03-08 21:03:48

标签: scheme eval racket

我想要以下代码:

(define fn.str "(lambda (x) (displayln x)")
(define fn.callable (string->lambda fn.str))
; and then 2 next lines should be valid
(fn.callable 123)
(apply fn.callable '(321))

但我最好能得到的是eval string并获得输出 我的目标是获得实际持有值(在这种情况下为λ)。

一般来说,这也是期望的效果:

(define val.num "500")
(define val.li "'(1 2 3)")
(define val.fn "(λ (a) a)")
(define num (string->value val.num)) ; => num is now holding 500
(define li (string->value val.li)) ; => list '(1 2 3)
(define fn (string->value val.fn)) ; => callable lambda

也许我已经足够接近解决方案,但它闻起来很难闻。

更多信息:

  • 这些字符串从外部进入程序(文件/用户输入)。
  • 提取/解析的值应绑定到某些内容(例如哈希表),以便进一步访问。

1 个答案:

答案 0 :(得分:2)

你对某些事情是正确的 - 从字符串中评估代码是一种糟糕的代码味道......也许有更好的方法来完成你想要完成的事情。无论如何,这是一个可能的解决方案:

(define-namespace-anchor a)
(define ns (namespace-anchor->namespace a))

(define (eval-string str)
  (eval (call-with-input-string str read) ns))

它按照样本输入的要求工作:

(define fn.str "(lambda (x) (displayln x))")
(define fn.callable (eval-string fn.str))
(fn.callable 123)
=> 123

(apply fn.callable '(321))
=> 321

(define val.num "500")
(eval-string val.num)
=> 500

(define val.li "'(1 2 3)")
(eval-string val.li)
=> '(1 2 3)

(define val.fn "(λ (a) a)")
(eval-string val.fn)
=> #<procedure>