要检测命令是否仍然存在,请使用info commands并检查它返回的列表的长度。(我假设您可以保证命令名称中没有 glob 字符;如果这是真的,那么检查非常便宜。)
while {[llength [info commands cor]]} {
puts [cor]
}
但是,如果您打算将协程用作生成器并且仅在循环中使用,则可以让它直接发出信号表明它已完成。为此,让它产生一个break.
coroutine cor apply {{} {
yield [info coroutine]; # A sensible result from [coroutine]
yield 1
yield 2
yield 3
return -code break
# [tailcall break] would also work
}}
while true {
puts [cor]
}
诀窍是你不要 yield休息;您将其用作协程的最终结果。(您的旧协程正在使用yieldas that 的结果,可能主要是靠运气。)
在写这篇文章时,我注意到了一个错误。运行协程的循环在 8.6.8(我检查过的唯一版本)的顶层不起作用。不知道为什么。在过程/apply上下文中很好(将协程名称作为参数传递是很自然的):
apply {c {
while true {
puts [$c]
}
}} cor
这些不同运行方式的字节码看起来并没有明显的不同。强制解释也会使事情正常进行,这很好地表明这是一个真正的错误。
set while while
$while true {
puts [cor]
}