1
Fork 0
mirror of https://git.savannah.gnu.org/git/guile.git synced 2025-05-11 00:00:49 +02:00
guile/module/slib/objdoc.txi
2001-04-14 11:24:45 +00:00

238 lines
7.5 KiB
Text

@code{(require 'object)}
@ftindex object
This is the Macroless Object System written by Wade Humeniuk
(whumeniu@@datap.ca). Conceptual Tributes: @ref{Yasos}, MacScheme's
%object, CLOS, Lack of R4RS macros.
@subsection Concepts
@table @asis
@item OBJECT
An object is an ordered association-list (by @code{eq?}) of methods
(procedures). Methods can be added (@code{make-method!}), deleted
(@code{unmake-method!}) and retrieved (@code{get-method}). Objects may
inherit methods from other objects. The object binds to the environment
it was created in, allowing closures to be used to hide private
procedures and data.
@item GENERIC-METHOD
A generic-method associates (in terms of @code{eq?}) object's method.
This allows scheme function style to be used for objects. The calling
scheme for using a generic method is @code{(generic-method object param1
param2 ...)}.
@item METHOD
A method is a procedure that exists in the object. To use a method
get-method must be called to look-up the method. Generic methods
implement the get-method functionality. Methods may be added to an
object associated with any scheme obj in terms of eq?
@item GENERIC-PREDICATE
A generic method that returns a boolean value for any scheme obj.
@item PREDICATE
A object's method asscociated with a generic-predicate. Returns
@code{#t}.
@end table
@subsection Procedures
@defun make-object ancestor @dots{}
Returns an object. Current object implementation is a tagged vector.
@var{ancestor}s are optional and must be objects in terms of object?.
@var{ancestor}s methods are included in the object. Multiple
@var{ancestor}s might associate the same generic-method with a method.
In this case the method of the @var{ancestor} first appearing in the
list is the one returned by @code{get-method}.
@end defun
@defun object? obj
Returns boolean value whether @var{obj} was created by make-object.
@end defun
@defun make-generic-method exception-procedure
Returns a procedure which be associated with an object's methods. If
@var{exception-procedure} is specified then it is used to process
non-objects.
@end defun
@defun make-generic-predicate
Returns a boolean procedure for any scheme object.
@end defun
@defun make-method! object generic-method method
Associates @var{method} to the @var{generic-method} in the object. The
@var{method} overrides any previous association with the
@var{generic-method} within the object. Using @code{unmake-method!}
will restore the object's previous association with the
@var{generic-method}. @var{method} must be a procedure.
@end defun
@defun make-predicate! object generic-preciate
Makes a predicate method associated with the @var{generic-predicate}.
@end defun
@defun unmake-method! object generic-method
Removes an object's association with a @var{generic-method} .
@end defun
@defun get-method object generic-method
Returns the object's method associated (if any) with the
@var{generic-method}. If no associated method exists an error is
flagged.
@end defun
@subsection Examples
@example
(require 'object)
@ftindex object
(define instantiate (make-generic-method))
(define (make-instance-object . ancestors)
(define self (apply make-object
(map (lambda (obj) (instantiate obj)) ancestors)))
(make-method! self instantiate (lambda (self) self))
self)
(define who (make-generic-method))
(define imigrate! (make-generic-method))
(define emigrate! (make-generic-method))
(define describe (make-generic-method))
(define name (make-generic-method))
(define address (make-generic-method))
(define members (make-generic-method))
(define society
(let ()
(define self (make-instance-object))
(define population '())
(make-method! self imigrate!
(lambda (new-person)
(if (not (eq? new-person self))
(set! population (cons new-person population)))))
(make-method! self emigrate!
(lambda (person)
(if (not (eq? person self))
(set! population
(comlist:remove-if (lambda (member)
(eq? member person))
population)))))
(make-method! self describe
(lambda (self)
(map (lambda (person) (describe person)) population)))
(make-method! self who
(lambda (self) (map (lambda (person) (name person))
population)))
(make-method! self members (lambda (self) population))
self))
(define (make-person %name %address)
(define self (make-instance-object society))
(make-method! self name (lambda (self) %name))
(make-method! self address (lambda (self) %address))
(make-method! self who (lambda (self) (name self)))
(make-method! self instantiate
(lambda (self)
(make-person (string-append (name self) "-son-of")
%address)))
(make-method! self describe
(lambda (self) (list (name self) (address self))))
(imigrate! self)
self)
@end example
@subsubsection Inverter Documentation
Inheritance:
@lisp
<inverter>::(<number> <description>)
@end lisp
Generic-methods
@lisp
<inverter>::value @result{} <number>::value
<inverter>::set-value! @result{} <number>::set-value!
<inverter>::describe @result{} <description>::describe
<inverter>::help
<inverter>::invert
<inverter>::inverter?
@end lisp
@subsubsection Number Documention
Inheritance
@lisp
<number>::()
@end lisp
Slots
@lisp
<number>::<x>
@end lisp
Generic Methods
@lisp
<number>::value
<number>::set-value!
@end lisp
@subsubsection Inverter code
@example
(require 'object)
@ftindex object
(define value (make-generic-method (lambda (val) val)))
(define set-value! (make-generic-method))
(define invert (make-generic-method
(lambda (val)
(if (number? val)
(/ 1 val)
(error "Method not supported:" val)))))
(define noop (make-generic-method))
(define inverter? (make-generic-predicate))
(define describe (make-generic-method))
(define help (make-generic-method))
(define (make-number x)
(define self (make-object))
(make-method! self value (lambda (this) x))
(make-method! self set-value!
(lambda (this new-value) (set! x new-value)))
self)
(define (make-description str)
(define self (make-object))
(make-method! self describe (lambda (this) str))
(make-method! self help (lambda (this) "Help not available"))
self)
(define (make-inverter)
(let* ((self (make-object
(make-number 1)
(make-description "A number which can be inverted")))
(<value> (get-method self value)))
(make-method! self invert (lambda (self) (/ 1 (<value> self))))
(make-predicate! self inverter?)
(unmake-method! self help)
(make-method! self help
(lambda (self)
(display "Inverter Methods:") (newline)
(display " (value inverter) ==> n") (newline)))
self))
;;;; Try it out
(define invert! (make-generic-method))
(define x (make-inverter))
(make-method! x invert! (lambda (x) (set-value! x (/ 1 (value x)))))
(value x) @result{} 1
(set-value! x 33) @result{} undefined
(invert! x) @result{} undefined
(value x) @result{} 1/33
(unmake-method! x invert!) @result{} undefined
(invert! x) @error{} ERROR: Method not supported: x
@end example