12.11 Syntax Utilities
(require racket/syntax) | package: base |
12.11.1 Creating formatted identifiers
函数
(format-id lctx fmt v ... [ #:source src #:props props #:cert ignored]) → identifier? lctx : (or/c syntax? #f) fmt : string? v : (or/c string? symbol? identifier? keyword? char? number?) src : (or/c syntax? #f) = #f props : (or/c syntax? #f) = #f ignored : (or/c syntax? #f) = #f
The format string must use only ~a placeholders. Identifiers in the argument list are automatically converted to symbols.
> (define-syntax (make-pred stx) (syntax-case stx () [(make-pred name) (format-id #'name "~a?" (syntax-e #'name))])) > (make-pred pair) #<procedure:pair?>
> (make-pred none-such) none-such?: undefined;
cannot reference an identifier before its definition
in module: top-level
internal name: none-such?
> (define-syntax (better-make-pred stx) (syntax-case stx () [(better-make-pred name) (format-id #'name #:source #'name "~a?" (syntax-e #'name))])) > (better-make-pred none-such) none-such?: undefined;
cannot reference an identifier before its definition
in module: top-level
internal name: none-such?
(Scribble doesn’t show it, but the DrRacket pinpoints the location of the second error but not of the first.)
函数
(format-symbol fmt v ...) → symbol?
fmt : string? v : (or/c string? symbol? identifier? keyword? char? number?)
> (format-symbol "make-~a" 'triple) 'make-triple
12.11.2 Pattern variables
语法
(define/with-syntax pattern stx-expr)
stx-expr : syntax?
> (define/with-syntax (px ...) #'(a b c)) > (define/with-syntax (tmp ...) (generate-temporaries #'(px ...))) > #'([tmp px] ...) #<syntax:eval:11:0 ((a9 a) (b10 b) (c11 c))>
> (define/with-syntax name #'Alice) > #'(hello name) #<syntax:eval:13:0 (hello Alice)>
12.11.3 Error reporting
parameter
(current-syntax-context stx) → void? stx : (or/c syntax? false/c)
函数
(wrong-syntax stx format-string v ...) → any
stx : syntax? format-string : string? v : any/c
> (wrong-syntax #'here "expected ~s" 'there) eval:14:0: ?: expected there
at: here
> (parameterize ([current-syntax-context #'(look over here)]) (wrong-syntax #'here "expected ~s" 'there)) eval:15:0: look: expected there
at: here
in: (look over here)
(define-syntax (my-macro stx) (parameterize ([current-syntax-context stx]) (syntax-case stx () __)))
12.11.4 Recording disappeared uses
parameter
→ (or/c (listof identifier?) false/c) (current-recorded-disappeared-uses ids) → void? ids : (or/c (listof identifier?) false/c)
语法
(with-disappeared-uses body-expr ... stx-expr)
stx-expr : syntax?
修改于package base的6.5.0.7版本:Added the option to include body-exprs.
函数
(syntax-local-value/record id predicate) → any/c
id : identifier? predicate : (-> any/c boolean?)
函数
(record-disappeared-uses id) → void?
id : (or/c identifier? (listof identifier?))
If not used within the extent of a with-disappeared-uses form or similar, has no effect.
修改于package base的6.5.0.7版本:Added the option to pass a single identifier instead of requiring a list.
12.11.5 Miscellaneous utilities
函数
(generate-temporary [name-base]) → identifier?
name-base : any/c = 'g
函数
(internal-definition-context-apply intdef-ctx stx) → syntax? intdef-ctx : internal-definition-context? stx : syntax?
函数
(syntax-local-eval stx [intdef-ctx]) → any
stx : syntax?
intdef-ctx :
(or/c internal-definition-context? (listof internal-definition-context?) #f) = '()
> (define-syntax (show-me stx) (syntax-case stx () [(show-me expr) (begin (printf "at compile time produces ~s\n" (syntax-local-eval #'expr)) #'(printf "at run time produces ~s\n" expr))])) > (show-me (+ 2 5))
at compile time produces 7
at run time produces 7
> (define-for-syntax fruit 'apple) > (define fruit 'pear) > (show-me fruit)
at compile time produces apple
at run time produces pear
修改于package base的6.90.0.27版本:Changed intdef-ctx to accept a list of internal-definition contexts in addition to a single internal-definition context or #f.
语法
(with-syntax* ([pattern stx-expr] ...) body ...+)
stx-expr : syntax?
> (with-syntax* ([(x y) (list #'val1 #'val2)] [nest #'((x) (y))]) #'nest) #<syntax:eval:21:0 ((val1) (val2))>