-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathcloneable.rkt
58 lines (49 loc) · 1.51 KB
/
cloneable.rkt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
#lang racket
;;; File:
;;; cloneable.rkt
;;; Summary:
;;; Tools for dealing with cloneable objects
;;; Author:
;;; Samuel A. Rebelsky
(require racket/generic)
(require "sstruct.rkt")
(provide (all-defined-out))
; +---------------+--------------------------------------------------
; | The interface |
; +---------------+
;;; (cloneable? val) -> boolean?
;;; val : any?
;;; Determine if val is cloneable. You can call `clone` on cloneable values.
(define-generics cloneable
(clone cloneable)
#:fallbacks
[(define clone
(lambda (val)
(error 'clone "~a is not cloneable" val)))])
; +-----------------------+------------------------------------------
; | A deprecated approach |
; +-----------------------+
;;; (struct-cloneable name ...) -> (void?)
;;; name : identifier?
;;; Make a cloneable structure.
;;;
;;; Deprecated.
(define-syntax struct-cloneable
(syntax-rules ()
[(struct-cloneable name rest ...)
(struct name rest ...
#:methods gen:cloneable
[(define clone
(lambda (val)
(struct-copy name val)))])]))
; +---------------------+--------------------------------------------
; | The #:cloneable tag |
; +---------------------+
;;; (permit-cloneable) -> (void)
;;; Permit the "cloneable" tag.
(define-syntax-rule (permit-cloneable)
(set-sstruct-tag! #:cloneable 0
(lambda (name)
`(#:methods gen:cloneable
[(define clone
(lambda (val) (struct-copy ,name val)))]))))