我正在对来自 Tcl8.6 和 Rivet 的 TclOO 进行一些试验,但我遇到了麻烦,因为我无法做我想做的事。
使用文件中的以下代码可以简单地重现该问题.rvt
:
<?
proc dumbproc {} {
puts "this is dumbproc ([namespace current])"
}
oo::class create foo {
method bar {} {
puts "this is bar ([namespace current])"
dumbproc
}
}
set obj [foo new]
dumbproc
$obj bar
如果我只是看一下代码,它似乎应该按预期工作,但实际上并没有,因为 Rivet 包的微妙行为和选择的特定配置。
在此示例中,我使用了一个.rvt
文件,其代码在::request
命名空间内执行,因此该dumbproc
过程的完全限定名称是::request::dumbproc
. 当在方法内部调用名称解析算法时bar
,它先搜索dumbproc
inside ::oo::Obj12
,然后搜索 in ::oo
,最后搜索 in ::
,但没有找到并给出以下错误。
this is dumbproc (::request) this is bar (::oo::Obj16)
invalid command name "dumbproc"
while executing
"dumbproc"
(class "::request::foo" method "bar" line 3)
invoked from within
"$obj bar"
(in namespace eval "::request" script line 21)
invoked from within
"namespace eval request {
puts -nonewline ""
proc dumbproc {} {
puts "this is dumbproc ([namespace current])"
}
oo::class create..."
所以,Tcl做它所做的事情是正确的,那就是一个特性。但是这种行为是不可预测的,因为当你编写一些类代码时,你必须知道它将被使用的上下文。
事实上,如果我放弃启动<?
Rivet 魔法,将代码放在test.tcl
文件中并在交互式会话中使用它,我会得到同样的错误:
$ tclsh
% namespace eval ::myns {source test.tcl}
this is dumbproc (::myns)
this is bar (::oo::Obj12)
invalid command name "dumbproc"
我试图通过将当前命名空间添加到类创建代码来解决这个问题
::oo::class create [namespace current]::foo { ... }
然后,我还尝试obj
在命名空间内创建对象
::oo::class create [namespace current]::foo { ... }
namespace eval [namespace current] {set obj [[namespace current]::foo new]}
然后,我切换到create
类的方法,为对象提供一个包含命名空间的限定名称
foo create [namespace current]::obj
obj bar
但一切都没有成功。每次试验都表明,无论我怎么做,TclOO 类中的方法总是在其对象唯一命名空间内执行。我错了吗?
有没有办法得到我想要的?TclOO 是否不打算以这种方式工作,在这种情况下为什么呢?真正让我惊讶的是这种依赖于上下文的行为,我不确定这是不是正确的事情,但也许我完全错了,并且有一些合理的案例,我错过了。