返回介绍

2.2 用宏实现 Self

发布于 2025-02-20 00:17:06 字数 1688 浏览 0 评论 0 收藏 0

在我们的 OBJECT 宏中使用上述模式:

(defmac (OBJECT ([field fname init] ...)
                ([method mname args body] ...))
  #:keywords field method
  (letrec ([self
            (let ([fname init] ...)
              (let ([methods (list (cons 'mname (λ args body)) ...)])
                (λ (msg . vals)
                  (apply (cdr (assoc msg methods)) vals))))])
    self))

(defmac (-> o m arg ...)
  (o 'm arg ...))

用一些点对象试试:

(define (make-point init-x)
  (OBJECT
   ([field x init-x])
   ([method x? () x]
    [method x! (nx) (set! x nx)]
    [method greater (other-point)
            (if (> (-> other-point x?) x)
                other-point
                self)])))

> (let ([p1 (make-point 5)]
        [p2 (make-point 2)])
    (-> p1 greater p2))
self: undefined;
 cannot reference undefined identifier

什么??我们明明用 letrec 定义了 self,为什么报错说它没有定义呢?原因是—— 卫生 !要知道 Scheme 的 syntax-rules 是卫生的,因此,它会透明地重命名宏引入的所有标识符,以确保在宏展开后他们不会意外绑定或者被绑定。使用 DrRacket 的宏步进器(macro stepper)可以很清楚地观察到这一点。你会看到,greater 方法中的 self 标识符与 letrec 表达式中的同名标识符的颜色不同。

幸运的是,defmac 支持一种方法,指定宏本身引入的标识符也可以被用户代码使用。这里我们唯一需要做的是指定 self 就是这样的标识符:

(defmac (OBJECT ([field fname init] ...)
                ([method mname args body] ...))
  #:keywords field method
  #:captures self
  (letrec ([self
            (let ([fname init] ...)
              (let ([methods (list (cons 'mname (λ args body)) ...)])
                (λ (msg . vals)
                  (apply (cdr (assoc msg methods)) vals))))])
    self))

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据
    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文