[comp.lang.lisp.x] How are superclass methods invoked?

J.K.Wight@newcastle.ac.uk (Jim Wight) (07/06/90)

I am relatively new to OOP and xlisp and am having some difficulty
getting to grips with the combination as the documentation is either
misleading or doesn't explain things in sufficient detail to match my
naivety.

For instance, the documentation states that:-

  "A message can also be sent from the body of a method by using the
   current object, but the method lookup starts with the object's
   superclass rather than its class."
   
I understand this to mean that if class1 is derived from Object, and
class2 from class1, and if each class has a method :whatever, say, then
class2's :whatever can include the expression (send self :whatever) in
order to invoke class1's :whatever. But this is not what happens. I
find that class2's :whatever recursively calls itself. Is this a bug in
the documentation or a bug in xlisp? 

Also, what is :sendsuper all about and how is it used?  It doesn't
appear to be a method of Object like :show or :class, which means that
I have to provide one through (send class2 :answer :sendsuper ...).
My difficulty is that I don't see what to put in its body. If I am able
to send a message to the superclass from within the class's :sendsuper
method then what does :sendsuper gain me?  Because of what was said in
the previous paragraph it can't include (send self :whatever ...)
unless class2 doesn't have a :whatever method, in which case I might
as well just say (send object2 :whatever) directly since class1's
:whatever method will be found through inheritance anyway (object2 is
an instance of class2).

I've obviously completely missed the point somewhere along the line,
so I would be grateful if someone would put me right. A simple example
of the use of :sendsuper would be appreciated.

I am using Version 2.0. If there is a more recent one could someone
provide the name of an ftp site.

Jim
---
Computing Laboratory            VOICE: +44 91 222 8238  FAX: +44 91 222 8232
University of Newcastle upon Tyne    JANET: J.K.Wight@uk.ac.newcastle
Newcastle upon Tyne               UUCP: ...!ukc!newcastle.ac.uk!J.K.Wight
NE1 7RU,  UK                 ARPA: J.K.Wight%uk.ac.newcastle@nsfnet-relay.ac.uk

tim@hpfcbig.SDE.HP.COM (Tim Mikkelsen) (07/16/90)

Since I wasn't able to mail directly, here is a reposting of the objects
primer......

Tim M
-----------------------------------------------------------------------------










			   XLISP 2.0 OBJECTS PRIMER


				     by 

			       Tim I Mikkelsen


			      February 3, 1990








	Copyright  (c) 1990 by Tim I.  Mikkelsen.  All Rights  Reserved.
	No part of this document may be copied, reproduced or translated
	for commercial use without prior written  consent of the author.
	Permission  is granted  for  non-commercial  use as long as this
	notice is left intact.


	________________________________________________________________

	
	One of the  features  in the design of XLISP is  object-oriented
	programming.  This  primer is  intended to serve as a very brief
	introduction  to the object  facilities of the XLISP 2.0 dialect
	of LISP.  Note that the object  features  of XLISP are not based
	on other existing object definitions in other LISP dialects.  If
	you find problems in the primer, I'd appreciate hearing.


			Tim Mikkelsen
			4316 Picadilly Drive
			Fort Collins, Colorado  80526



PROGRAMMING STYLES
______________________________________________________________________________


There are many  programming  paradigms  (models).  Some of the paradigms
are procedural, functional, rule-based, declarative and object-oriented.
A language can have aspects of one or many of these programming  models.


Procedure-Oriented

The programming paradigm most people are familiar with is the procedural
style.  The primitives in procedural  programming  are:  subroutines and
data  structures.  Through  these  primitives,   programmers  have  some
limited abilities to share programs and program fragments.  C and Pascal
are examples of procedural  languages.  Some procedural  languages (such
as Modula and ADA) have  extensions  that provide for better  sharing of
code.


Object-Oriented Programming

Object-oriented  programming  is based  on the  primitives  of  objects,
classes and messages.  Objects are defined in terms of classes.  Actions
occur by sending a message to an object.  An object's  definition can be
inherited  from  more  general  classes.  Objective-C  and C++ both  are
object-oriented  dialects of the C language.  Many dialects of LISP have
some object oriented extension (Flavors, Common LOOPS, CLOS and others).
There  currently is standards  work  proceeding  to add  object-oriented
programming to Common LISP.


Object Oriented Programming

So, the  object-oriented  programming model is based around the concepts
of objects,  classes and messages.  An object is essentially a black box
that contains internal state  information.  You send an object a message
which causes the object to perform some  operation.  Objects are defined
and described through classes.

One aspect of an object is that you do not have to know what is inside -
or how it  works - to be able to use it.  From a  programming  point  of
view,  this is very  handy.  You can  develop a series  of  objects  for
someone to use.  If you need to change what goes on inside, the users of
the objects should be unaware.

Another aspect of objects is that of  inheritance.  You can build up new
classes  from  existing  classes  by  inheriting  the  existing  class's
functionality  and then extending the new  definition.  For example, you
can  define a tool class  (with  various  attributes)  and then go about
creating  object  instances  tool-1,  tool-2,  and so on.  You can  also
create new sub-classes of the tool class like  power-tool.  This is also
very handy because you don't have to  re-implement  something if you can
build it up from existing code.




XLISP OBJECT-ORIENTED PROGRAMMING
______________________________________________________________________________



XLISP OBJECT TERMINOLOGY

There  are, as  previously  mentioned,  many  different  languages  with
object-oriented  extensions and facilities.  The terminology, operations
and  styles of these are very  different.  Some of the main  definitions
for XLISP's object-oriented extensions are:

	Object data type        The OBJECT DATA TYPE is a built-in  data
				type of  XLISP.  Members  of the  object
				data  type  are  object   instances  and
				classes.

	Object instances        An  OBJECT   INSTANCE  is  a   composite
				structure that contains  internal  state
				information,  methods  (the  code  which
				respond to  messages),  a pointer to the
				object  instance's  defining class and a
				pointer  to  the  object's  super-class.
				XLISP   contains   no  built-in   object
				instances.

	Class objects           A  CLASS  OBJECT  is,  essentially,  the
				template for defining the derived object
				instances.  A  class  object,   although
				used  differently  from a simple  object
				instance,  is  structurally  a member of
				the  object   data   type.  It  is  also
				contains  the  linking   mechanism  that
				allows  you to build  class  hierarchies
				(sub-classes and  super-classes).  XLISP
				contains  two  built-in  class  objects:
				OBJECT and CLASS.

	Message selector        The MESSAGE  SELECTOR is the symbol that
				is used to  select a  particular  action
				(Method) from the object.

	Message                 The  MESSAGE is the  combination  of the
				message  selector  and the data (if any)
				to be sent to the object.

	Method                  The METHOD is the actual  code that gets
				executed  when the object  receives  the
				Message.



SENDING MESSAGES

The  mechanism  for  sending  messages to XLISP  objects is via the SEND
function.  It takes an object, a message  selector and various  optional
arguments (depending on the message selector).

The way that a user  creates a new object is to send a :NEW message to a
previously  defined  class.  The  result  of this  SEND will  return  an
object, so this is normally preceded by a SETQ.  The values shown in the
examples  that follow may not match what you see if you try this on your
version of XLISP - this is not an error.  The  screens  that are used in
the various examples are similar to what you should see on your computer
screen.  The ">" is the normal XLISP prompt (the  characters that follow
the prompt is what you should type in to try these examples).


	 ________________________________________________________________
	|
	|	> (setq my-object (send object :new))
	|	#<Object: #2e100>
	|________________________________________________________________


The object  created here is of limited  value.  Most often, you create a
class  object and then you create  instances  of that  class.  So in the
following  example, a class called MY-CLASS is created that inherits its
definition from the a built-in CLASS definition.  Then two instances are
created of the new class.

	 ________________________________________________________________
	|
	|	> (setq my-class (send class :new '()))
	|	#<Object: #27756>
	|	
	|	> (setq my-instance (send my-class :new))
	|	#<Object: #27652>
	|
	|	> (setq another-instance (send my-class :new))
	|#<Object: #275da>
	|________________________________________________________________



CLASSES

Previously,  a :NEW  message was used to create an object.  The  message
used to see what is in an object is the :SHOW message.

	 ________________________________________________________________
	|
	|	> (send my-class :show)
	|	Object is #<Object: #27756>, Class is #<Object: #23fe2>
	|	  MESSAGES = NIL
	|	  IVARS = NIL
	|	  CVARS = NIL
	|	  CVALS = NIL
	|	  SUPERCLASS = #<Object: #23fd8>
	|	  IVARCNT = 0
	|	  IVARTOTAL = 0
	|	#<Object: #27756>
	|________________________________________________________________


From the display of the MY-CLASS  object you can see there are a variety
of components.  The components of a class are:

	Class Pointer           This  pointer  shows to what  class  the
				object (instance or class) belongs.  For
				a  class,  this  always  points  to  the
				built-in  object  CLASS.  This  is  also
				true  of the  CLASS  object,  its  class
				pointer points to itself.

	Superclass Pointer      This  pointer  shows what the next class
				up the class  hierarchy is.  If the user
				does  not  specify  what  class  is  the
				superclass,   it  will   point   to  the
				built-in class OBJECT.

	Messages                This  component  shows what messages are
				allowed   for   the   class,   and   the
				description  of the method  that will be
				used.  If the method is  system-defined,
				it will show up in the form of '#<Subr-:
				#18b98>'.  Remember   that   the   class
				hierarchy    (through   the   Superclass
				Pointer) is  searched  if the  requested
				message is not found in the class.

	Instance Variables      This   component   lists  what  instance
				variables will be created when an object
				instance is created.  If no instances of
				the class  exist,  there are no Instance
				Variables.  If there are 5 instances  of
				a  class,  there  are  5  complete   and
				different   groups   of   the   Instance
				Variables.

	Class Variables         The  CLASS  VARIABLES  (CVAR)  component
	and Values		lists what class variables  exist within
				the  class.  The  Class  Values   (CVAL)
				component  shows what the current values
				of the variables  are.  Class  Variables
				are used to hold state information about
				a class.  There  will be |Bone of each|A
				of the Class  Variables,  independent of
				the  number of  instances  of the  class
				created.

A BETTER EXAMPLE

The  example  previously  shown  does work, but the class and  instances
created  don't really do anything of  interest.  The  following  example
sets up a tool class and creates some tool instances.

	 ________________________________________________________________
	|
	|	> (setq my-tools (send class :new '(power moveable operation)))
	|	#<Object: #277a6>
	|
	|	> (send my-tools :answer :isnew '(pow mov op) 
	|					 '((setq power pow)
	|					   (setq moveable mov)
	|					   (setq operation op)))
	|	#<Object: #277a6>
	|
	|	> (setq drill (send my-tools :new 'AC t 'holes))
	|	#<Object: #2ddbc>
	|
	|	> (setq hand-saw (send my-tools :new 'none t 'cuts))
	|	#<Object: #2dc40>
	|
	|	> (setq table-saw (send my-tools :new 'AC nil 'cuts))
	|	#<Object: #2db00>
	|________________________________________________________________


So, a class of objects called MY-TOOLS was created.  Note that the class
object  MY-TOOLS was created by sending the :NEW message to the built-in
CLASS  object.  Within  the  MY-TOOL  class,  there are three  instances
called DRILL, HAND-SAW and TABLE-SAW.  These were created by sending the
:NEW message to the MY-TOOLS  class object.  Notice that the  parameters
followed the message selector.


INSTANCES

The  following  is a display of the  contents of some of the  previously
created instances:

	 ________________________________________________________________
	|
	|	> (send drill :show)
	|	Object is #<Object: #2ddbc>, Class is #<Object: #277a6>
	|	  POWER = AC
	|	  MOVEABLE = T
	|	  OPERATION = HOLES
	|	#<Object: #2ddbc>
	|
	|	> (send hand-saw :show)
	|	Object is #<Object: #2dc40>, Class is #<Object: #277a6>
	|	  POWER = NONE
	|	  MOVEABLE = T
	|	  OPERATION = CUTS
	|	#<Object: #2dc40>
	|________________________________________________________________


From  the  display  of  these  instances  you  can see  there  are  some
components and values.  The components of an instance are:

	Class Pointer           This  pointer  shows to which  class the
				current object instance  belongs.  It is
				through this link that the system  finds
				the methods to execute for the  received
				messages.

	Instance Variables      The Instance  Variables (IVAR) component
	and Values		lists what  variables  exist  within the
				instance.  The Instance Values component
				holds  what the  current  values  of the
				variables  are.  Instance  Variables are
				used to hold state  information for each
				instance.  There  will  be  a  group  of
				Instance Variables for each instance.




METHODS

There have been a few of the messages and methods in XLISP shown to this
point (:NEW and :SHOW).  The following are the methods built into XLISP:

	:ANSWER         The  :ANSWER  method  allows  you to  define  or
			change methods within a class.

	:CLASS 		The :CLASS method returns the class of an object.

	:ISNEW          The :ISNEW method  causes an instance to run its
			initialization  code.  When the :ISNEW method is
			run on a class, it resets the class state.  This
			allows  you  to  re-define  instance  variables,
			class variables, etc.

	:NEW            The :NEW method allows you to create an instance
			when the :NEW message is sent to a  user-defined
			class.  The :NEW  method  allows you to create a
			new class (when the :NEW  message is sent to the
			built-in CLASS).

	:SHOW 		The :SHOW method displays the instance or class.



SENDING MESSAGES TO A SUPERCLASS

In  addition  to the SEND  function,  there is another  function  called
SEND-SUPER.  The SEND-SUPER  function causes the specified message to be
performed  by the  superclass  method.  This  is a  mechanism  to  allow
chaining of methods in a class hierarchy.  This chaining behavior can be
achieved  by  creating a method  for a class with the  :ANSWER  message.
Within the body of the  method,  you  include a  SEND-SUPER  form.  This
function is allowed only inside the  execution of a method of an object.


OBJECT AND CLASS

The definition of the built-in class OBJECT is:

	 ________________________________________________________________
	|
	|	> (send object :show)
	|	Object is #<Object: #23fd8>, Class is #<Object: #23fe2>
	|	  MESSAGES = ((:SHOW . #<Subr-: #23db2>) 
	|	  	      (:CLASS . #<Subr-: #23dee>) 
	|		      (:ISNEW . #<Subr-: #23e2a>))
	|	  IVARS = NIL
	|	  CVARS = NIL
	|	  CVALS = NIL
	|	  SUPERCLASS = NIL
	|	  IVARCNT = 0
	|	  IVARTOTAL = 0
	|	#<Object: #23fd8>
	|________________________________________________________________


Note that OBJECT is a class - as opposed to an "instance-style"  object.
OBJECT has no superclass, it is the top or root of the class  hierarchy.
OBJECT's class is CLASS.

	 ________________________________________________________________
	|
	|	> (send class :show)
	|	Object is #<Object: #23fe2>, Class is #<Object: #23fe2>
	|	  MESSAGES = ((:ANSWER . #<Subr-: #23e48>) 
	|	  	      (:ISNEW . #<Subr-: #23e84>) 
	|		      (:NEW . #<Subr-: #23ea2>))
	|	  IVARS = (MESSAGES IVARS CVARS CVALS SUPERCLASS 
	|		   IVARCNT IVARTOTAL)
	|	  CVARS = NIL
	|	  CVALS = NIL
	|	  SUPERCLASS = #<Object: #23fd8>
	|	  IVARCNT = 7
	|	  IVARTOTAL = 7
	|	#<Object: #23fe2>
	|________________________________________________________________


CLASS has a superclass of OBJECT.  It's class is itself - CLASS. 



A MORE REALISTIC EXAMPLE 
______________________________________________________________________________


The following is an example, using the idea of tools again.  It contains
a hierarchy of tool  classes.  The top of the class  hierarchy is TOOLS.
HAND-TOOLS and SHOP-TOOLS are sub-classes of TOOLS.  The example creates
instances of these  sub-classes.  It is possible to extend this  example
in various ways.  One obvious  extension would be to create a third tier
of classes  under  HAND-TOOLS  that could  contain  classes like drills,
screwdrivers, pliers and so on.




;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;
;	File name:	tools.lsp
;	Author:		Tim Mikkelsen
;	Description:	Object-oriented example program
;	Language:	XLISP 2.0
;
;	Date Created:	10-Jan-1988
;	Date Updated:	2-Apr-1989
;	
;	(c) Copyright 1988, by Tim Mikkelsen, all rights reserved.
;	    Permission is granted for unrestricted non-commercial use.
;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;

;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;
;	Define the superclasses and classes
;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;

;
; make TOOLS superclass
;	with a different :ISNEW method
;	added methods are :BORROW and :RETURN
;	class variables are	NUMBER		contains # of tool instances
;				ACTIVE-LIST	contains list of current objects
;	instance variables are 	POWER 		list - (AC BATTERY HAND)
;				MOVEABLE 	CAN-CARRY or CAN-ROLL or FIXED
;				OPERATIONS	list
;				MATERIAL 	list - (WOOD METAL PLASTIC ...)
;				PIECES 		list
;				LOCATION	HOME or person's name
;

(setq tools (send class :new '(power 
			       moveable 
			       operations 
			       material 
			       pieces 
			       location) 
			     '(number active-list)))
(send tools :answer :isnew '() 
		           '((if (null number) (setq number 1)
		      			       (setq number (1+ number)))
			     (setq active-list (cons self active-list))
			     (setq location 'home)))
(send tools :answer :borrow '(by-who)
		            '((if (eq location 'home) (setq location by-who)
		       				      (print "you can't"))))
(send tools :answer :return '()
		            '((if (eq location 'home) (print "got it already")
		       				      (setq location 'home))))

;
; make HAND-TOOLS class
;	with a different :ISNEW method
;	new instance variable	WEIGHT		<number> of pounds
;	the rest is inherited from TOOLS 
; 

(setq hand-tools (send class :new '(weight) '() tools))
(send hand-tools :answer :isnew '(pow op mat parts w-in)
			        '((setq power pow)
			          (setq moveable 'can-carry)
			          (setq operations op)
			          (setq material mat)
    			          (setq pieces parts)
			          (setq weight w-in)
			          (send-super :isnew)))

;
; make SHOP-TOOLS class
;	with a different :ISNEW method
;	no new instance variables
;	the rest is inherited from TOOLS 
; 

(setq shop-tools (send class :new '() '() tools))
(send shop-tools :answer :isnew '(pow mov op mat parts)
			        '((setq power pow)
			         (setq moveable mov)
			         (setq operations op)
			         (setq material mat)
			         (setq pieces parts)
			         (send-super :isnew)))

;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;
;	Create instances of various tool classes 
;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;

(setq hand-drill (send hand-tools :new 		; make an instance - HAND-DRILL
			     '(ac) 
			     '(drill polish grind screw)
			     '(wood metal plastic)
			     '(drill drill-bits screw-bits buffer)
			     '2.5))

(setq table-saw (send shop-tools :new 		; make an instance - TABLE-SAW
			     '(ac)
			     'fixed
			     '(rip cross-cut)
			     '(wood plastic)
			     '(saw blades fence)))


(setq radial-arm (send shop-tools :new 		; make an instance = RADIAL-ARM
			     '(ac)
			     'can-roll
			     '(rip cross-cut)
			     '(wood plastic)
			     '(saw blades dust-bag)))


The following  session shows how to use the tool  definitions  from this
better  example.  The example starts at the OS shell and brings up xlisp
running the file 'tools.lsp'.

	 ________________________________________________________________
	|
	|	cmd? xlisp tools
	|	; loading "init.lsp"
	|	; loading "tools.lsp"
	|	> (send hand-drill :borrow 'fred)
	|	FRED
	|
	|	> (send table-saw :return)
	|	"got it already"
	|	"got it already"
	|
	|	> (send hand-drill :borrow 'joe)
	|	"you can't"
	|	"you can't"
	|
	|	> (send hand-drill :return)
	|	HOME
	|________________________________________________________________


So, Fred was able to borrow the HAND-DRILL.  When an attempt was made to
return the  TABLE-SAW,  it was  already  at home.  A second  attempt  to
borrow the HAND-DRILL  indicated that "you can't" because it was already
lent out.  Lastly, the HAND-DRILL was returned successfully.  (Note that
the "got it  already"  and "you  can't"  strings  show up  twice  in the
display because the methods both print and return the string.)

The following session shows the structure of the TOOLS object:

	 ________________________________________________________________
	|
	|	> (send tools :show)
	|	Object is #<Object: #276fc>, Class is #<Object: #23fe2>
	|	  MESSAGES = ((:RETURN . #<Closure-:RETURN: #2dbd0>) 
	|	  	      (:BORROW . #<Closure-:BORROW: #2ddba>) 
	|		      (:ISNEW . #<Closure-:ISNEW: #274a4>))
	|	  IVARS = (POWER MOVEABLE OPERATIONS MATERIAL PIECES LOCATION)
	|	  CVARS = (NUMBER ACTIVE-LIST)
	|	  CVALS = #(3 (#<Object: #2cadc> 
	|	  	       #<Object: #2cda2> 
	|	       #<Object: #2d0e0>))
	|	  SUPERCLASS = #<Object: #23fd8>
	|	  IVARCNT = 6
	|	  IVARTOTAL = 6
	|	#<Object: #276fc>
	|________________________________________________________________


The two TOOLS sub-classes HAND-TOOLS and SHOP-TOOLS structure looks like:

	 ________________________________________________________________
	|
	|	> (send hand-tools :show)
	|	Object is #<Object: #2dab8>, Class is #<Object: #23fe2>
	|	  MESSAGES = ((:ISNEW . #<Closure-:ISNEW: #2d7a2>))
	|	  IVARS = (WEIGHT)
	|	  CVARS = NIL
	|	  CVALS = NIL
	|	  SUPERCLASS = #<Object: #276fc>
	|	  IVARCNT = 1
	|	  IVARTOTAL = 7
	|	#<Object: #2dab8>
	|
	|	> (send shop-tools :show)
	|	Object is #<Object: #2d680>, Class is #<Object: #23fe2>
	|	  MESSAGES = ((:ISNEW . #<Closure-:ISNEW: #2d450>))
	|	  IVARS = NIL
	|	  CVARS = NIL
	|	  CVALS = NIL
	|	  SUPERCLASS = #<Object: #276fc>
	|	  IVARCNT = 0
	|	  IVARTOTAL = 6
	|	#<Object: #2d680>
	|________________________________________________________________


The class HAND-TOOLS has an instance HAND-DRILL which looks like:

	 ________________________________________________________________
	|
	|	> (send hand-drill :show)
	|	Object is #<Object: #2d0e0>, Class is #<Object: #2dab8>
	|	  WEIGHT = 2.5
	|	  POWER = (AC)
	|	  MOVEABLE = CAN-CARRY
	|	  OPERATIONS = (DRILL POLISH GRIND SCREW)
	|	  MATERIAL = (WOOD METAL PLASTIC)
	|	  PIECES = (DRILL DRILL-BITS SCREW-BITS BUFFER)
	|	  LOCATION = HOME
	|	#<Object: #2d0e0>
	|________________________________________________________________