我的意思是为什么我们不能只使用已经收到的变量作为过程中的参数。
你可以。它只是变得烦人。
通常,当您将变量的 name 传递给命令时,命令可以修改该变量。这方面的经典示例是 set 和 incr 命令,它们都将变量名称作为第一个参数。
set thisVariable $thisValue
您也可以对过程执行此操作,但是当变量是在过程调用者的上下文中定义的变量时,您需要从过程的上下文访问该变量,该变量可能是命名空间或可能是一个不同的局部变量框架。为此,我们通常使用upvar,它将一个局部变量的别名设置为另一个上下文中的变量。
例如,这里是incr的重新实现:
proc myIncr {variable {increment 1}} {
upvar 1 $variable v
set v [expr {$v + $increment}]
}
为什么写入局部变量v 会导致调用者上下文中的变量被更新?因为我们给它起了别名(在内部,它是通过一个指向另一个变量的存储结构的指针来设置的;一旦完成upvar,它就会非常快)。 global和variable使用相同的底层机制;它们都归结为快速变量别名。
如果你使用uplevel 代替,你可以不这样做,但这会变得相当烦人:
proc myIncr {variable {increment 1}} {
set v [uplevel 1 [list set $variable]]
set v [expr {$v + $increment}]
uplevel 1 [list set $variable $v]
}
太恶心了!
或者,假设我们根本没有这样做。然后我们需要通过它的值传递变量,然后分配结果:
proc myIncr {v {increment 1}} {
set v [expr {$v + $increment}]
return $v
}
# Called like this
set foo [myIncr $foo]
有时是正确的,但工作方式完全不同!
Tcl 的核心原则之一是,您可以使用标准库命令(例如 if 或 puts 或 incr)执行几乎所有您自己编写的命令。没有关键字。自然可能存在一些效率问题,并且某些命令可能需要用另一种语言(例如 C)完成才能正常工作,但语义不会使任何命令特别。它们全部只是简单的命令。