Hi!
I have discovered some interesting change in 1.1.1 that I could not find in ChangeLog.
In 1.1.1 one can access to subobject as method call Example b::a can be produced as b a My problem is that I have "a" as method and as subobject from b. Can I force "b" to invoke method "a" and not return subobject "a"?
Artur Trzewik
Example: % [artur@localhost xotclIDE]$ /opt/tcl845/bin/tclsh8.4 % package require XOTcl 1.1 % namespace import xotcl::* % Class A ::A % Class B ::B % B create b ::b % A create b::a ::b::a % b a ::b::a % [artur@localhost xotclIDE]$ /opt/tcl844/bin/tclsh8.4 % package require XOTcl 1.0 % namespace import xotcl::* % Class A ::A % Class B ::B % B create b ::b % A create a ::a % b a b: unable to dispatch method 'a' %
On Tuesday 30 December 2003 12:25, Artur Trzewik wrote:
Hi!
I have discovered some interesting change in 1.1.1 that I could not find in ChangeLog.
Artur, you discovered this change not in version 1.1.1, but in the pre-release of version 1.1.2, that i have sent to you a few days ago. The change is documented in the Changelog of 1.1.2
In 1.1.1 one can access to subobject as method call Example b::a can be produced as b a My problem is that I have "a" as method and as subobject from b.
There are several rationales for this:
* Background: In general Tcl distinguishes between procs and commands, where the fist ones are defined in tcl and the others in C. If you define a proc/command, then you are overwriting potentially a command/proc.
Objects and classes are commands in xotcl. Every Object builds a namespace (not necessarily a tcl-namespace) in xotcl , in which the procs, subcommands and subobjects of the object are stored.
The situation in versions up to 1.1.1 was a follows
Object o o proc a {} {puts "i am proc a"} o a ;# prints i am proc a Object o::a o a ;# error message: o: unable to dispatch method 'a'
The created Object has overwritten the proc a, since it occupies slot "a" in the command table of object o. The same happens as well when a proc is overwriting an object, the same situation happens as well with other subcommands (not only procs) of the object.
So the situation is quite bad, since the sub-object - can delete procs/commands - it cannot be called directly - it breaks in invariant: an (sub)object is a (sub)command and can be called exactly like it.
There are several useful scenarios for the new behavior. * one can use xotcl objects to define commands with subcommands, where the object forms the "main command" and the procs/commands define the subcommands. For example one can defined a specialized version of Tcl's file-command (say ns::file) like in the following:
namespace eval ns { Object file file proc tail name { set tail [file tail $name] regexp {[^/\]+$} $tail tail return $tail } file proc unknown {subcmd args} { return eval ::file $subcmd $args } } The neat thing is that it is possible - to redefine sub-commands via procs (instead of a large and hard-to-extend switch statement) - to use interceptors (filters/mxins) for certain subcommands (maybe you are only interested in "file delete") - to extend command on the fly by defining new subcommands - to use the unknown mechanism to produce dynamic error message about valid subcommands.
This was possible since a long time in XOTcl; what's new in the prerelease of 1.1.2 is that the same can be used now with sub-subcommands and so on in a uniform way. This makes it possible, for example, to use this mechanism for xotcl's info command as well, which is already a subcommand of the object/class (e.g. [Object info instances].
* In many cases the programs become syntactically cleaner than before. One can also write now instead of
Http::Wrk instproc init .... { ... [self]::connection translation binary .... the invocation of the subobject without [self] by:
Http::Wrk instproc init .... { ... my connection translation binary ....
Interesting enough, in many situations the new variant is faster as well (Artur, check out speedtest.xotcl of the 1.1.2 prerelease).
Can I force "b" to invoke method "a" and not return subobject "a"?
The calling order is mixins before instmixins before procs/cmds before instprocs from the classes. Therefore, the object a shadows in your example the instprocs, which is the problem you are facing.
well, we have currently following options implemented:
a) if you call an object by it name (without method), it returns the name of the object. e.g.:
Class C C instproc a {} {puts instproc-a} C o; C o::a puts [o::a] ;# outputs ::o::a
This is implemented via a method "defaultmethod",
puts [o::a defaultmethod] ;# outputs ::o::a
which can be redefined by the user (e.g. via proc). This method can call other things as needed and return finally [self]. you can not do is to call the instproc a of class C via next (since the active object is alreday ::o::a).
You can, however call from defaultmethod ::xotcl::classes::C::a directly, but this is not recommended, since you will have the "wrong" self, etc.
b) you can define an interceptor for method a, e.g. via mixin class
c) use different nameing conventions for sub-objects, e.g. C o; C o::@a
Sorry for being long, but i hope this explains the rational for the change.
best regards and all the best in the new year
-gustaf neumann
Hallo!
The new convention of calling subobject is indeed interesting. I have looked in my code and discovered a lot of [self]::subobject method ... To write it as my subobject method must be more performant because [self]::subobject will be every time substituted and can be not handled as internal Tcl object.
Yet XOTclIDE is not compatible with this convention because I use Class subobjects as descriptors for methods with the same name as method. This subobjects are used to handle version control informations. And in the one case a get subobject instead of method call.
c) use different nameing conventions for sub-objects, e.g. C o; C o::@a
This is for me the best solution. In the best case I need to change only one method.
The another Aspect in this new feature is the code analyse (syntax checker). In the old code by call my method "method" can be only method. Now it can be also subobject that can be defined dynamically. Static syntax analyse is so not very reliable. But also a human who need to read a code must consider the two cases. Convention or another syntax can be good in this case.
[self]::b method my b method; # what is object, what is method name, what is argument [my subobject b] method; # alternative but not good for subcommands my . b method; # . were special syntax for it or build all subobjects with special prefix my @b method
There are many pro and contra. It is not clear for me now.
Happy new year Artur Trzewik