Each class has a parent class which can be used as a container for bits of code that apply to a more general class of objects. In this section we show how this mechanism works in detail.
We begin by creating a new type of basic list.
i1 : X = new Type of BasicList |
i2 : parent X |
The parent of X is BasicList, as desired, thus methods applicable to basic lists will also apply also to instances of X. One such method is the method for creating a net from a basic list; here is its code:
i3 : code(net,BasicList) |
This code is run automatically to display an instance of X, so if we make one, we'll be able to see what it is:
i4 : x = new X from {2,3,4} |
Now let's imagine we wish to treat instances of X as vectors, and to negate one by negating its entries. As it happens, no method for this has been installed for basic lists, as we can check with lookup.
i5 : lookup(symbol -, X) === null |
We install and test a new method as described in installing methods.
i6 : - X := t -> apply(t,i -> -i); |
i7 : - x |
This method will apply automatically to subclasses of X, as we see now.
i8 : Y = new Type of X; |
i9 : y = new Y from {4,5,6} |
i10 : - y |
For binary methods, there is an apparent ambiguity in deciding exactly how inheritance will work. Let's illustrate by making a new subclass Z of X.
i11 : Z = new Type of X; |
i12 : z = new Z from {7,8,9} |
Now let's install two methods, either of which might conceivably be applied to evaluate the expression y+z, and see what happens.
i13 : Y + X := (a,b) -> XY; |
i14 : X + Z := (a,b) -> ZX; |
i15 : y + z |
The result is the symbol XY. The reason is that after finding that no method applies directly for adding an instance of Y to an instance of Z, the search continues: Z is replaced by its parent X, and so on. (After enough unsuccessful iterations of this, the second type is reset to Z, the first type is replaced by its parent, and the search continues.)
The same search order applies to method functions defined with method.