Why does Tcl not come with a sleep command? Why, it just doesn't call it sleep. It calls it after.
The TclX package includes a sleep command (with syntax similar to the POSIX shell command).
Puts the process (or at least the current thread) to sleep for seconds seconds (truncated to an integer). Note that the OS may wait for longer than requested before waking the process up.
The Expect package's sleep is equivalent.
Note that Tcl's built-in after command uses delay units of milliseconds whereas the TclX/Expect command works with seconds (i.e., a factor of 1000 different). Be careful when converting.
However, Jeff Hobbs notes in a discussion held elsewhere:
[someone proposed the code ]
proc sleep {time} { after $time set end 1 vwait end }
[and Jeff responds:]
This [the above code] is actually not correct because vwait must receive a globally scoped variable, so add global end above (or rather use a less common global name).
This has advantages over regular after $time because it does allow other events to continue processing, but that difference should be understood.
Also, when using the form of after that takes a command in Tcl without Tk present, you have to make sure that vwait is called (or update), to kick in the event loop.
This was a horribly misguided thing... dangerously appealing and guaranteed to create snake balls.
I tried to remove the code below from this page, but some evil genius put it back!
Will suspend evaluation of current code block, but allow the event loop to continue handling pending events.
This particular version of uniqkey is my favorite because the keys generated will generally sort in time created order.
proc uniqkey { } { set key [ expr { pow(2,31) + [ clock clicks ] } ] set key [ string range $key end-8 end-3 ] set key [ clock seconds ]$key return $key } proc sleep { ms } { set uniq [ uniqkey ] set ::__sleep__tmp__$uniq 0 after $ms set ::__sleep__tmp__$uniq 1 vwait ::__sleep__tmp__$uniq unset ::__sleep__tmp__$uniq }
Example:
after 4000 puts foo! sleep 5000 puts bar!
DKF: Provided you have an event loop already going and you're in a Tcl 8.6 coroutine, you can do an event-handling sleep without the reentrancy troubles:
proc sleep {ms} { after $ms [list [info coroutine]] yield }
GPS: You swilly wabbits. :)
$ tclsh8.4 % interp alias {} sleep {} after sleep % sleep 400 % interp alias {} wait {} sleep wait % wait 800
after has problems with fine-grained delays, at least on my system; its probably related to unix timeslicing.
% time {after 1} 100 10006 microseconds per iteration
So asking for a 1ms delay gets you something closer to 10ms. I'm in need of something much closer to accurate, so I came up with two variants of a busywaiting delay. They watch clock clicks to see when the appropriate amount of time has passed. The first, delay-bw is a pure busy-wait, while delay-ev uses the event loop.
namespace eval delay { variable _i variable c/ms proc calibrate {} { variable c/ms puts "calibrating clock clicks.." set start [clock clicks] after 1000 set end [clock clicks] set c/ms [expr {($end-$start)/1000.0}] puts "speed: [expr {${c/ms}*1000}] clicks per second" } calibrate # busywaiting delay proc delay-bw {sec} { variable c/ms set s [clock clicks] while {[clock clicks] < $s+(${c/ms}*$sec)} { # do nothing } # busywaiting "after idle" delay, using event loop proc delay-ev {sec} { variable c/ms set s [clock clicks] set e [expr {$s+$sec*${c/ms}}] evwait ::delay::_i $e vwait ::delay::_i unset ::delay::_i } # worker for delay-ev # continually reschedules itself via "after idle" until end time proc evwait {var {end 0}} { set ct [clock clicks] if {$ct < $end} { after idle [list ::delay::evwait $var $end] return } else { set $var 0 } } }
I get much more accurate delays using these:
% time {::delay::delay-ev 1} 1000 1283 microseconds per iteration % time {::delay::delay-bw 1} 1000 1225 microseconds per iteration
JR