summaryrefslogtreecommitdiffstats
path: root/txr.1
diff options
context:
space:
mode:
Diffstat (limited to 'txr.1')
-rw-r--r--txr.1240
1 files changed, 240 insertions, 0 deletions
diff --git a/txr.1 b/txr.1
index 46cfe27b..81440eb9 100644
--- a/txr.1
+++ b/txr.1
@@ -28196,6 +28196,23 @@ compares the replacement object in place of the original, and an
hash table uses the replacement object as the key for the purposes of
hashing and comparison.
+.NP* Custom Slot Expansion
+
+The
+.code defstruct
+macro has a provision for for application-defined clauses, which may
+be defined using the
+.code define-struct-clause
+macro. This macro associates new clause keywords with custom expansion.
+The
+.code :delegate
+clause of
+.code defstruct
+is in fact implemented externally to
+.code defstruct
+using
+.codn define-struct-clause .
+
.coNP Macro @ defstruct
.synb
.mets (defstruct >> { name | >> ( name << arg *)} < super
@@ -30803,6 +30820,229 @@ if there are no forms. The invocations of
.code call-finalizers
take place just before the value of the last form is returned.
+.coNP Macro @ define-struct-clause
+.synb
+.mets (define-struct-clause < keyword < params <> [ body-form ]*)
+.syne
+.desc
+The
+.code define-struct-clause
+macro makes available a new, application-defined
+.code defstruct
+clause. The clause is named by
+.metn keyword ,
+which must be a keyword symbol, and is implemented as a macro
+transformation by the
+.meta params
+and
+.metn body-form s
+of the definition. The definition established by
+.code define-struct-clause
+is called a
+.IR "struct clause macro" .
+
+A struct clause macro is invoked when
+.code defstruct
+syntax is processed which contains one or more clauses which are
+headed by the matching
+.meta keyword
+symbol.
+
+The
+.meta params
+comprise a macro-style parameter list which must match the
+invoking clause, otherwise an error exception is thrown.
+When
+.meta params
+successfully matches the clause parameters, the parameters
+are destructured into the parameters and the
+.metn body-form s
+are evaluated in the scope of those parameters.
+
+The
+.metn body-form s
+must return a possibly list of
+.code defstruct
+clauses, not a single clause.
+
+Each of the returned clauses is examined for the possibility that
+it may be a struct clause macro; if so, it is expanded.
+
+The built-in clause keywords
+.codn :static ,
+.codn :instance ,
+.codn :function ,
+.codn :method ,
+.codn :init ,
+.code :postinit
+and
+.code :fini
+may not be used as the names of a struct clause macro; if any of these
+symbols is used as the
+.meta keyword
+parameter of
+.codn define-struct-clause ,
+an error exception is thrown.
+
+The return value of a
+.code define-struct-clause
+macro invocation is the
+.meta keyword
+argument.
+
+.TP* Examples:
+
+.verb
+ ;; Trivial struct clause macro which consumes any number of
+ ;; arguments and produces no slots:
+
+ (define-struct-clause :nothing (. ignored-args))
+
+ ;; Consequently, the following defines a struct with one slot, x:
+ ;; The (:nothing ...) clause disappears by producing no clauses.
+
+ (defstruct foo ()
+ (:nothing 1 2 3 beeblebrox)
+ x)
+
+ ;; struct clause macro called :multi which takes an initial value
+ ;; and zero or more slot names. It produces instance slot definitions
+ ;; which all use that same initial value.
+
+ (define-struct-clause :multi (init-val . names)
+ (mapcar (lop list init-val) names))
+
+ ;; define a struct with three slots initialized to zero:
+
+ (defstruct bar ()
+ (:multi 0 a b c)) ;; expands to (a 0) (b 0) (c 0)
+
+ ;; struct clause macro to define a slot along with a
+ ;; get and set method.
+
+ (define-struct-clause :getset (slot getter setter : init-val)
+ ^((,slot ,init-val)
+ (:method ,getter (obj) obj.,slot)
+ (:method ,setter (obj new) (set obj.,slot new))))
+
+ ;; Example use:
+
+ (defstruct point ()
+ (:getset x get-x set-x 0)
+ (:getset y get-y set-y 0))
+
+ ;; This has exactly the same effect as the following defstruct:
+
+ (defstruct point ()
+ (x 0)
+ (y 0)
+ (:method get-x (obj) obj.x)
+ (:method set-x (ob new) (set obj.x new))
+ (:method get-y (obj) obj.y)
+ (:method set-y (ob new) (set obj.y new)))
+.brev
+
+.coNP Struct clause macro @ :delegate
+.synb
+.mets (:delegate < name <> ( param +) < delegate-expr <> [ target-name ])
+.syne
+.desc
+The
+:delegate
+struct clause macro provides a way to define a method which is implemented entirely
+by delegation to a different object. The name of the method is
+.meta name
+and its parameter list is specified in the sme way as in the
+.meta :method
+clause. Instead of a method body, the
+.code :delegate
+clause has an expression
+.meta delegate-expr
+and an optional
+.meta target-name
+which defaults to
+.metn name .
+The
+.meta delegate-expr
+must be an expression which the delegate method can evaluate to
+produce a delegate object. The delegate method then passes its
+arguments to the target method, given by the
+.meta target-name
+argument, invoked on the delegate object. If the delegate method has optional
+parameters which have not received an argument value, those parameters
+are treated as if they had received the colon symbol
+.code :
+as their value, and that value is passed on. If the delegate method has
+variadic parameters, they are applied to the target. If optional parameters
+specified in
+.code :delegate
+are given argument values, those are discarded, playing no role in the
+delegation.
+
+.TP* Example:
+
+Structure definitions:
+
+.verb
+ (defstruct worker ()
+ name
+ (:method work (me)
+ `worker @{me.name} works`)
+ (:method relax (me : (min 15))
+ `worker @{me.name} relaxes for @min min`))
+
+ ;; "contractor" class has a sub ("subcontractor") slot
+ ;; which is another contractor of the same type.
+ ;; The subcontractor's own sub slot, however is going
+ ;; to be a worker.
+
+ (defstruct contractor ()
+ sub
+ (:delegate work (me) me.sub.sub)
+ (:delegate break (me : min) me.sub.sub relax))
+.brev
+
+The
+.code contractor
+structure's
+.code work
+and
+.code break
+methods delegate to the sub-subcontractor, which is going to be
+instantiated as a
+.code worker
+object. Note that the
+.code break
+method delegates to a differently named method
+.codn relax .
+
+.verb
+ ;; The objects are set up as described above.
+ ;; general contractor co has a co.sub subcontractor,
+ ;; and co.sub.sub is a worker:
+
+ (defvar co (new contractor
+ sub (new contractor
+ sub (new worker name "foo"))))
+
+ ;; Call work method on general contractor:
+ ;; this invokes co.sub.sub.(work) on the worker.
+
+ co.(work) -> "worker foo works"
+
+ ;; Call break method on general contractor with
+ ;; no argument. This causes co.sub.sub.(relax :)
+ ;; to be invoked, triggering argument defaulting:
+
+ co.(break) -> "worker foo relaxes for 15 min"
+
+ ;; Call break method with argument. This
+ ;; invokes co.sub.sub.(relax 5), specifying a
+ ;; value for the default argument:
+
+ co.(break 5) -> "worker foo relaxes for 5 min"
+.brev
+
.SS* Special Structure Functions
Special structure functions are user-defined methods or structure functions