In January of 2004, a poster to comp.lang.tcl asked which of the
tcl extensions providing
object orientation allowed one to define class members which were other objects.
Several people submitted answers - and some of the answers provided demonstrated how the different extensions might do just that.
Below, examples for
Snit,
itcl,
stooop (
moodss related),
XOtcl and
TclOO are provided.
Are there other popular Tcl object oriented extensions to represent?
snit::type C2 {
variable this
variable that
constructor {args} {
set this [C2 %AUTO%]
set that [C2 %AUTO%]
}
destructor {
catch {$this destroy}
catch {$that destroy}
}
method doThis {} {
$this doSomething
}
method doThat {} {
$that doSomething
}
}
Shouldn't it be like this:
snit::type C1 {
option -partof
method doSomething {} {
puts "[$self cget -partof]'s $self doin' it"
}
}
snit::type C2 {
delegate method doThis to this as doSomething
delegate method doThat to that as doSomething
constructor {args} {
install this using C1 %AUTO% -partof $self
install that using C1 %AUTO% -partof $self
$self configurelist $args
}
destructor {
catch {$this destroy}
catch {$that destroy}
}
}
itcl::class C1 {
}
itcl::class C2 {
variable m1
variable m2
constructor {} {
set m1 [C1 #auto]
set m2 [C1 #auto]
}
destructor {
itcl::delete object $m1 $m2
}
}
class C2 {
proc C2 {this} {
set ($this,o1) [new C1]
set ($this,o2) [new C1]
}
proc ~C2 {this} {
delete $($this,o1) $($this,o2)
}
class C1 {
proc C1 {this} {}
proc ~C1 {this} {}
}
}
delete [new C2]
Note: in the example above, C1 is a class embedded in C2, but might as well be defined outside of C2, as in:
class C1 {
proc C1 {this} {}
proc ~C1 {this} {}
}
class C2 {
proc C2 {this} {
set ($this,o1) [new C1]
set ($this,o2) [new C1]
}
proc ~C2 {this} {
delete $($this,o1) $($this,o2)
}
}
There are some posibilities
Example 1 (generate global objects in the constructor and refer to it via instance variables)
Class C1
Class C2
C2 instproc init {} {
my instvar o1 o2
set o1 [C1 new]
set o2 [C1 new]
}
C2 instproc destroy {} {
my instvar o1 o2
$o1 destroy
$o2 destroy
next
}
Example 2 (by using anonymous nested objects. Does not need to destroy explicit the sub-objects)
Sub-objects are better to implement aggregation or membership (part of, has a)
Class C1
Class C2 -parameter {o1 o2}
C2 instproc init {} {
my o1 [C1 new -childof [self]]
my o2 [C1 new -childof [self]]
}
Example 3 (by using named sub-objects. Does not need to destroy explicit the sub-objects)
Quite similar to C++ object members
Class C1
Class C2
C2 instproc init {} {
C1 create [self]::o1
C1 create [self]::o2
}
Example 4 (by using anonymous sub-objects generated via parameters. Sub-objects are destroyed automatically when container is deleted)
Quite similar to example 2, but no need for a explicit constructor
::xotcl::Class::Parameter C1
Class C2 -parameter {
{o1 -Class C1 -default 1}
{o2 -Class C1 -default 1}
}
(By
DKF on 22-Jan-2011)
oo::class create C1
oo::class create C2 {
variable m1 m2
constructor {} {
set m1 [C1 new]
set m2 [C1 new]
}
destructor {
$m1 destroy
$m2 destroy
}
}
Or alternatively:
oo::class create C1
oo::class create C2 {
variable m1 m2
constructor {} {
set m1 [C1 create m1]
set m2 [C1 create m2]
}
}
Note that in the latter example, the
full name of m1 is
[info object namespace $c2inst]::m1; this means it will be
automatically destroyed when the outer container (
$c2inst) is destroyed due to the deletion of the instance
namespace. This is tricky (and is a trick used by
TDBC to manage its cleanup). Also note that inside the methods of C2, the subobjects can be referred to as
m1 and
m2 because the names match the variable names. However the local names are formally arbitrary; all that's needed is for them to be in the right namespace (and it's probably a bad idea to overwrite or override any existing Tcl command).