Ключ для ответа datum->syntax
. Основная идея заключается в том, что вы хотите взять некоторые случайные данные и превратить их в синтаксис - в этом случае превратить символ в идентификатор. Идентификатор - это в основном символ с некоторой лексической информацией, которая (очень приблизительно) указывает, как он связан. Используя datum->syntax
, вы можете сделать именно это: он ожидает существующий фрагмент синтаксиса, из которого копирует привязку, и элемент данных (здесь символ), который является значением, содержащимся в оболочке синтаксиса.
Вот пример, который демонстрирует define-struct
-подобный инструмент, использующий это:
#lang scheme
;; implements a defstruct-like macro that uses association lists
(define-syntax (defstruct-lite stx)
(syntax-case stx ()
[(defstruct-lite name field ...)
(let ([make-id
(lambda (template . ids)
(let ([str (apply format template (map syntax->datum ids))])
(datum->syntax stx (string->symbol str))))])
(with-syntax ([make-name (make-id "make-~a" #'name)]
[name? (make-id "~a?" #'name)]
[(arg ...) (generate-temporaries #'(field ...))]
[(name-field ...)
(map (lambda (f) (make-id "~a-~a" #'name f))
(syntax->list #'(field ...)))])
#'(begin
(define (make-name arg ...) (list 'name (cons 'field arg) ...))
(define (name? x) (and (pair? x) (eq? 'name (car x))))
(define (name-field x)
(and (name? x) (cdr (assq 'field (cdr x)))))
...)))]))
А вот пример его использования:
(defstruct-lite point x y)
(point-y (make-point 1 2))