13.5.1. Crash Course in Tcl
The Tcl language is easy to learn. If you are at all familiar with
other scripting languages, such as the Bourne or C shell, Tcl/Tk will pose
no threat to you.
For this reason, we will not spend a great deal of time on the Tcl
language itself. It is straightforward and can be learned with
the help of the various Tcl manual pages or
John Ousterhout's
excellent book, Tcl and the Tk Toolkit. This book
describes not only how to write Tcl and Tk scripts but also how to
use the Tcl/Tk libraries in your own applications.
Let's start with a simple example. The following Tcl script counts the
lines in the given filename:
1 #!/usr/bin/tclsh -f
2
3 if {$argc != 1} {
4 error "lc <filename>"
5 }
6
7 set thefile [open [lindex $argv 0] r]
8 set count 0
9
10 while {[gets $thefile line] >= 0} {
11 set count [expr $count + 1]
12 }
13
14 puts "Read $count lines."
Lines 3-5 use a simple
if statement to ensure that there is
one argument to the script--that being the filename containing lines
to count. The
if command takes two arguments: an expression
and a block of code to execute if the expression evaluates as true.
(Like C, zero indicates false, nonzero indicates true.)
Each of the arguments to the if command are contained in
braces, which group a set of words (or lines) together
as a single argument. Although this syntax may remind you of C
or Perl, Tcl's command-parsing behavior is actually quite simple.
For example, it allows a command argument (here, the body
of code on lines 3-5 containing the error command) to span
multiple lines only if the opening brace is at the end of a line.
If we had written the if command as:
if {$argc != 1}
{ error "lc <filename>" }
Tcl would have given us the error:
Error: wrong # args: no script following "$argc != 1" argument
wrong # args: no script following "$argc != 1" argument
while executing
"if {$argc != 1} "
(file "./lc.tcl" line 3)
In other words, Tcl doesn't know that the second argument to
if is on the following line.
The body of the if command, on line 4, uses the error
command to display an error message and exit the Tcl script.
On line 7, we open the file given as the first command-line argument,
and assign the resulting file pointer to the variable thefile.
The set command must be used to
assign values to variables. This is because all Tcl commands must
begin with a command name; we can't set the variable a to 1
using something like:
a = 1
because
a refers to a variable, not a command. Instead, we use:
set a 1
Later, to refer to the value of the variable
a, we would use
$a.
The first argument to set is the name of the variable to
assign, and the second argument is the value. Here, we have:
set thefile [open [lindex $argv 0] r]
Square brackets
[…] are used to specify a
subscript, a sequence of commands to nest within the current
command. The subscript is executed, and its return value substituted
in the subscript's place.
For example, let's look at the subscript:
open [lindex $argv 0] r
This script executes the
open command to open the file given
as its first argument. The second argument,
r, indicates
that the file should be opened for reading.
The first argument to open is the subscript:
lindex $argv 0
The
lindex command is used to index a list, or array, of items.
In this case, we wish to obtain the 0th element of the
$argv
array, which contains the command-line arguments to the program,
minus the command name itself. (This is unlike the use of
argv in C programs.) Therefore, the 0th element of
$argv
is the first command-line argument.
Let's say that we named our script lc.tcl, and
invoked it as:
eggplant$ lc.tcl /etc/passwd
Therefore, within the command:
set thefile [open [lindex $argv 0] r]
the nested subscript:
open [lindex $argv 0] r
will be replaced with:
open "/etc/passwd" r
which will, in turn, be replaced with the value of the file pointer
corresponding to
/etc/passwd. The net result is that the file
pointer value is assigned to the variable
thefile.
On line 8, we set the variable count to 0, which
acts as our line counter.
Lines 10-12 contain a simple while loop,
which repeatedly reads lines from the file until
end of file (EOF):
while {[gets $thefile line] >= 0} {
set count [expr $count + 1]
}
As we can see, the
while command takes two arguments: a
condition and a block of code to execute while the condition is
true. Here, the loop condition is:
[gets $thefile line] >= 0
We see the subscript:
gets $thefile line
which executes the
gets command. This reads a single line from
the file pointer
$thefile and assigns it to the variable
line.
gets returns the count of the number of characters
read, or -1 if
EOF is reached. Therefore, the
while loop will
continuously read lines from the file until
gets returns
a value less than zero.
The body of the while loop is:
set count [expr $count + 1]
which increments the value of
count. Remember that Tcl commands
must begin with a command name. Therefore, arithmetic expressions
are handled using the
expr command. Here, the
subscript:
expr $count + 1
returns the value of the variable
count plus 1. This is the
canonical way to increment variables within Tcl.
Finally, on line 14, we see:
puts "Read $count lines."
which uses the
puts command to display a string to standard output.
Here is a sample run of this script:
eggplant$ lc.tcl /etc/passwd
Read 144 lines.