Tcl/Tk快速入门

原贴:http://if.ustc.edu.cn/~xbzhou/blog/archives/tcl_cn/l-tcl/l-tcl-2-1.html


Tcl/Tk 的起源 第 1 页(共2 页)


Tcl 是“工具控制语言(Tool Control Language)”的缩写。Tk 是 Tcl“图形工具箱”的扩展,它提供各种标准的 GUI 接口项,以利于迅速进行高级应用程序开发。

John K. Ousterhout(见图像)于 1988 年开始开发 Tcl/Tk(读作“tickle tee-kay”),然后是加州大学伯克利分校(UCB)的一名教授继续对它进行开发。Tcl 是以可扩展性、短的学习曲线和易于嵌入为特定目标而设计的。Tk 的开发始于 1989 年,第一个版本于 1991 年问世。Ousterhout 博士在他离开 UCB 之后继续开发 Tcl/Tk,然后由于工作需要,他继续为 Sun Microsystems 公司工作。现在,也就是在写本文之时,他在 Scriptics(它开发出 Ajuba Solution,已由 Interwoven 收购)继续改进该语言,目前的稳定版本是 8.3.2,在写本文的时候,8.4 版本正在开发之中。

有关更多详细信息,请参阅“History of Tcl”页面。

工具和文件 第 2 页(共2 页)


在 Linux 系统上研究 Tcl/Tk,需要有两个主要程序。它们是 tclshwish。正如可以从其名称所辨别的那样,所以前者是 Tcl 外壳,常用于为外壳脚本提供执行环境。Wish 类似于 tclsh,它是针对窗口化的 GUI 环境。

输入下列命令,检查这些文件是否存在:



~/tcltk$ which tclsh

/usr/bin/tclsh



~/tcltk$ which wish

/usr/bin/wish



which 命令将返回指定可执行文件的路径。如果没有看到与这相似的结果,则需要转到 Scriptics Tcl/Tk 页面下 载并构建您自己的副本。当然,如果系统上不存在这些程序,并不表示有任何问题。不象 Perl,通常不会将 Tcl/Tk 视为是 Linux 操作的基本要素。我所知道的每个发行版都和某一版本的 Tcl/Tk 一起交付,最常见的扩展作为 CD-ROM 或在线资源库的一部分。从这些资源,这些工具通常很容易安装。如果有困难,请与 GNU/Linux 软件的发行商联系。


Tcl tick 的构成 第 1 页(共15 页)


在下面的清单中,将会发现第一个很常见的示例程序,它是用 Tcl 实现的。这是一个完整的脚本:第一行命令调用 tclsh 环境,第二行命令执行实际工作。用您所选择的文本编辑器创建该脚本,输入 chmod +x hello.tcl 使之成为可执行文件,然后执行它以测试您这件“作品”。



~/tcltk$ cat hello.tcl

#!/usr/bin/tclsh

puts stdout {Hello, World!}



~/tcltk$ ./hello.tcl

Hello, World!



Tcl 和 Tk 是解释型的、可扩展脚本语言。与 BSD 许可证十分相似,该许可证允许在任何情况下自由使用该软件,只要在所有副本中保留该版权并且在任何分发中一字不差地传递通告。这个许可证条款使 Tcl/Tk 成为自由软件。

Tcl/Tk 是一种解释型环境。可以通过添加预编译的 C 函数来扩展 Tcl 解释器,可从 Tcl 环境内部调用 Tcl 解释器。可以为特定目的或一般的以及广泛使用的而定制这些扩展。我们将在教程的后面看一些扩展并着重看一下第一个扩展 — 非常流行的 Expect

在接下来的几屏中,将回顾 Tcl 语言的一些主要特性,从元字符和全局变量到运算符、数学函数以及核心命令。毕竟,这些命令使 Tcl/Tk 成为有特色的,逐步发展的语言。请记住,在本教程中没有篇幅涉及每条命令。这里只突出一部分,以后您可以更进一步了解 Tcl/Tk。

 

 

 
 

#!/usr/bin/tclsh

# filename hello2.tcl

# This program code shows

# metacharacter usage



puts stdout "Hello, World! "a"

puts stdout {Hello, World! "a}



set Pints 6

set Days 7



puts stdout "The answer to the "

universe is [eval $Pints * $Days]!"n"



***

~/tcltk$ ./hello2.tcl

Hello, World!

Hello, World! "a

The answer to everything is 42!





 


Tcl 元字符 第 2 页(共15 页)


元字符是在 Tcl/Tk 环境的上下文中有特殊含意的字符或字符对,它们包括分组语句、封装字符串、终止语句以及其它,如下表所示。左边列出的代码中演示了一些元字符。要注意的一个特殊特性是,当使用花括号(防止替代和扩展)代替双引号时,输出中的差异。

字符 用作
# 注释
; newline 语句分隔符
Name 变量(区分大小写)
Name(idx) 数组变量
Name(j,k,l...) 多维数组
" string " 带替换的引用
{ string } 不带替换的引用
[ string ] 命令替换
" char 反斜杠替代
" 行继续(在行尾)

 

 

#!/usr/bin/tclsh

#

# Demonstrate global variables

# and backslash substitution



if {$argc >= 1} {

set N 1

foreach Arg $argv {

puts stdout "$N: $Arg"n"

set N [expr $N + 1]

if {$Arg == "ring"} {

puts stdout ""a"

}

}

} else {

puts stdout "$argv0 on "

X Display $env(DISPLAY)"n"

}



***

~/tcltk$ ./hello3.tcl

./hello3.tcl on X Display :0.0



~/tcltk$ ./hello3.tcl ring

1: ring



 


Tcl 全局变量和反斜杠替代 第 3 页(共15 页)


当 Tcl/Tk 脚本开始运行时,存在几个全局变量(如果在当前环境中为非空,则是预先定义的)。这些变量允许如下对操作系统进行访问:argc 是对脚本自变量的计数,而不是对调用的名称进行计数。 argv 是自变量的列表(不是数组)。argv0 是调用的文件名(可以是符号链接)。env 是根据当前外壳的环境变量名建立下标的数组。errorCode 存储有关最近的 Tcl 错误信息,errorInfo 包含对这同一个错误事件的堆栈跟踪。该列表还有另外 12 个 tcl_xxx 变量,从 tcl_interactivetcl_version。可以在 Tcl/Tk in a Nutshell 中找到好的总结,(有关更多信息,请参阅本教程末尾的“参考资料”)。

在左边的样本代码中,使用了其中几个变量以及(又一次)使用了一些反斜杠引用的字符("n 和 "a)。 " char 允许替代非打印 ASCII 字符。这对于 UNIX 下的许多脚本语言和外壳环境都是常见的。如表中说明的那样,对于没有定义替代的反斜杠引用的字符只被简单地回送到输出。

"字符 替代
"a 响铃
"b 退格
"f 换页
"n 或 "newline 新行
"r 回车
"t 水平制表
"v 垂直制表
"space ("" ") 空格
"ddd 八进制值
"xddd... 十六进制值
"c 回显‘c’
"" 反斜杠

 

 

 


~/tcltk$ cat maths.tcl

#!/usr/bin/tclsh

#

# Demonstrate operators and

# math functions



set PI [expr 2 * asin(1.0)]



if {$argc == 3} {

set X [lindex $argv 0]

set Y [lindex $argv 1]

set Rad [lindex $argv 2]



set Dist [expr sqrt(($X*$X)+($Y*$Y))]

set Cir [expr 2*$PI*$Rad]

set Area [expr $PI*$Rad*$Rad]



puts stdout "Distance = $Dist"

puts stdout "Circumference = $Cir"

puts stdout "Area = $Area"



} else {

puts stdout "Wrong argument count!"

puts stdout "Needs X, Y, and Radius"

}



********

~/tcltk$ ./maths.tcl 3 4 5

Distance = 5.0

Circumference = 31.4159265359

Area = 78.5398163397



 


Tcl 运算符和数学函数 第 4 页(共15 页)


Tcl 支持一组标准的运算符和数学函数。这些运算符包括算术、位和逻辑运算符,可以通过 expr 命令使用常规的运算符优先次序规则进行求值。另外,考虑到 Tcl 的实质是面向字符串的脚本语言,所以对一些数学函数进行了合理的补充:

 

  • 三角函数包括 cos(x)、acos(x)、cosh(x)、sin(x)、asin(x)、sinh(x)、tan(x)、atan(x)、atan2(y, x)、tanh(x) 和 hypot(x, y)。与这些函数相关的单位是弧度。
  • Log 函数是 exp(x)、log(x) 和 log10(x)。
  • 算术函数是 ceil(x)、floor(x)、fmod(x, y)、pow(x, y)、abs(x)、int(x)、double(x) 和 round(x)。
  • rand() 和 srand(x) 是处理随机数的函数。

 

左边这个示例使用了其中一些运算符和函数来计算指定点到原点之间的距离,并返回给定半径的圆的周长和面积。另外,在这个示例中,使用列表下标(lindex)命令来访问 $argv 的个别元素。


 

 

...

#

# parse command line switches

set Optimize 0

set Verbose 0



foreach Arg $argv {

switch -glob -- $Arg {

-o* {set Optimize 1}

-v* {set Verbose 1}

default {

error "Unknown $Arg"

}

}

}



set LineCount 0

while {[gets stdin Line] >= 0} {

# to confuse Vanna White...

Remove_Vowels $Line "

$Optimize $Verbose

incr LineCount

}

return LineCount

...





 


Tcl 中的循环和分支 第 5 页(共15 页)


Tcl 中的循环命令是 whileforforeach。条件(转移)命令是 if/then/else/elsifswitch。上述命令的限定语句是 breakcontinuereturnerror。最后,catch 命令提供了错误处理能力。

if/then/else/elsif 已在前面几屏中演示过。在正式语法中会用到 then,但通常会省略掉它。

在左边这个示例中,foreach 结构给 switch 命令提供命令行自变量。当处理自变量时(注意:不正确的输入会终止脚本,因为还没有实现处理错误的 catch),while 循环通过为每一行调用过程并同时对行计数器 1 来处理输入。代码段结束时,返回处理的行数。


 

 

~/tcltk$ tclsh

% set Phrase "hello, world!"

hello, world!



% string toupper $Phrase

HELLO, WORLD!



% string totitle $Phrase

Hello, world!



% string match ello $Phrase

0



% string match *ello* $Phrase

1



% string length $Phrase

14



% append Phrase "Nice day, eh?"

hello, world!

Nice day, eh?



% string toupper $Phrase

HELLO, WORLD!

NICE DAY, EH?



% string wordend $Phrase 7

12



 


Tcl 字符串和模式匹配 第 6 页(共15 页)


字符串是 Tcl 中的基本数据类型。string 命令实际上是一组命令,这些命令都是属于 string 的。在使用中,正如在左边的示例中所看到,string 读取的方式非常象来自 OOP 编程特定对象方法的应用程序。

表示信息的 string 命令是 lengthbytelength(可以有所不同,这取决于字符集)。返回布尔值(1 或 0)的比较是 compareequalmatch。这里的模式匹配是由“文件名替换”(简单类型的匹配通常与外壳操作相关)完成。还可以通过独特的 regex 和 regsub 命令来使用“高级正规表达式”。

在 Tcl 中执行 indexlastfirstwordendwordstart 命令可以实现下标功能。字符串修改是由 tolowertouppertotitletrimtrimlefttrimrightreplacemap 来处理的。后者需要预先定义一个字符映射表。用 range 抽取子字符串,用 repeat 多次输出字符串。

可以使用 append 命令,将文本添加到现有变量中。通过使用与 C 语言的 printf 命令相同的样式和约定,format 命令可用来生成输出字符串。scan 对字符串进行解析并将值赋值给变量。最后,从 Tcl 8.0 开始,用 binary formatbinary scan 命令添加了将二进制数据作为字符串处理的功能(因而能够处理空字符,而不会失败)。


 

~/tcltk$ tclsh

% set c1 {Bob Carol}

Bob Carol



% set c2 [list Ted Alice]

Ted Alice



% set Party1 [list $c1 $c2]

{Bob Carol} {Ted Alice}



% set Party2 [concat $c1 $c2]

Bob Carol Ted Alice



% linsert $Party1 1 Richard

{Bob Carol} Richard {Ted Alice}



%



 


Tcl 列表 第 7 页(共15 页)


列表在 Tcl 中有两个主要用途。我们已经在通过 foreach 命令(在 Tcl 中的循环和分支中找到)处理命令行自变量的环境中看到第一个用途。第二个用途是动态地构建 Tcl 命令的元素,可以在本教程后面看到使用 eval 命令来执行这种用途。

list 命令接受它的所有自变量并将它们返回在一个列表环境中。自变量可以是值或变量。在左边这个示例中,可以手工创建列表,或可将其它列表视作自变量来使用列表(从而保存第一个“Party”的两对方向)。或者,concat 命令用于将两个或多个列表合并到顶级项的单个实体,返回第二个更有趣的“Party”。

其它一些有用的列表命令及其语法是:

  • llength $List — 返回顶级项的计数结果。
  • lindex $List n — 返回已建立下标的项,从 0 开始计数。
  • lrange $List i j — 返回列表元素的范围。
  • lappend $List item... — 将项附加到列表。
  • linsert $List n item... — 在列表中的指定位置上插入一项或多项,向下移列表中的其它项。

 

其余列表命令还包括 lreplacelsearchlsortsplit 命令将字符串作为输入并生成经过正确解析的列表,并且在指定的字符处断开字符串。join 执行相反操作,接受列表元素并将它们串在一起,用 joinstring 分隔它们。


~/tcltk$ tclsh

% set People(friend) Tom

Tom

% set People(spouse) Marcia

Marcia

% set People(boss) Jack

Jack

% array names People

friend boss spouse

% set Person $People(friend)

Tom

% array get People

friend Tom boss Jack spouse Marcia

% set People(friend) "

[concat $People(friend) Bob]


Tom Bob

% set Person $People(friend)

Tom Bob

%



 


Tcl 数组 第 8 页(共15 页)


理解 Tcl 数组的捷径是,将它们视作与 Perl 散列相同的东西。Tcl 数组不是用数字建立下标的线性数据结构,除非选择对数据强加那种解释。尽管带空格的字符串需要用引号括起或需要一个变量引用,但下标(或键)可以是任何字符串。

正如一般的变量一样,使用 set 命令初始化数组,如左边所示。圆括号内是给出的下标部分。请注意,圆括号不象花括号或双引号那样提供分组。一旦初始化为数组,就不能将变量作为单一变量来访问。如左边列表底部所示,数组元素也可以是列表。


 

更多 Tcl 数组 第 9 页(共15 页)



变量作用域规则 第 11 页(共15 页)


作用域规则描述了过程和变量名以及值在程序的不同层次上的可见性。例如,在脚本的最外层定义的变量是全局变量。缺省情况下,全局变量是不可见的,在 过程内部也不可用它们的值。这允许过程的编写者自由地定义变量名并赋值,而不必担心覆盖对于局部作用域上未知的重要变量。要使全局变量在过程内部变得可 见,必须将它声明为在过程内,就象在上一屏的示例中,对 PI 使用 global 命令那样。

upvar 命令提供将局部变量与另一个作用域中变量的值相关联的设施。这允许根据名称将变量调用进过程,这对于当过程需要可以修改在另一个作用域的值而不仅仅使用它时,就显得非常方便。这个命令语法是 upvar level $VarName LocalVar ,其中 level 是到当前作用域之外的步骤数。“#0”表示全局作用域这一层。


#!/usr/bin/tclsh

#

# Demonstrate Data Structures

# using procedural wrappers



proc UserAdd { Acct rName eMail phone } {

global uData



if {[info exists uData($Acct,rname)]} {

return 1

}

set uData($Acct,rname) $rName

set uData($Acct,email) $eMail

set uData($Acct,phone) $phone

return 0

}



puts stdout [UserAdd bpb"

Brian bilbrey@junk.com 555-1212]



puts stdout [UserAdd tom"

Tom tom@junk.com 555-1212]



puts stdout [UserAdd bpb"

Brian bilbrey@junk.com 555-1212]



******

~/tcltk$ ./datas.tcl

0

0

1



 


Tcl 中的数据结构 第 12 页(共15 页)


除简单的多维数组以外,通常建议用已专门用于过程接口的数组来实现 Tcl 数据结构。从结构的使用者角度来看,虽然这种设计隐藏了具体实现细节,但提供了执行重要的错误检查能力。

左边这个示例中,在将 uData 声明为全局变量后,代码执行检查,以查看帐户是否已经不存在。如果存在,则过程返回(非零)错误消息。这个返回可以用于切换到生成错误文本输出。对于本 例,我们简单地提供三个连续输入,其中包括一次重复输入。这会产生如示例底部所示的输出,“1”表示由于一个重复的帐户名称而引起的一个错误返回,这是我 们有意这样做的。

其它可能的数据结构包括数组列表、已链接或双重链接的数组或其中的各种排列。因为 Tcl 8.0 所具有的列表重实现提供了不变的访问次数,所以数组列表结构相当有效。



~/tcltk$ tclsh

% file exists hello3.tcl

1

% file executable testit

0

% file pathtype ./hello3.tcl

relative

% set dir1 home

home

% set dir2 brian

brian

% set dir3 tcltk

tcltk

% file join /$dir1 dir2 dir3

/home/dir2/dir3

% file delete testit~

%



 


路径和文件 第 13 页(共15 页)


文件和路径操作是跨平台环境中具有挑战性的问题。对于主机 OS,Tcl 使用 UNIX 路径名(缺省情况下,用‘/’字符分隔)和本机路径名结构。即使当程序内的数据构造正确时,也很难确保用户输入与系统需求匹配。file join 命令用于将 UNIX 格式转换成本机路径名。其它路径字符串命令包括 file splitdirnamefile extensionnativenamepathtypetail

在它扮演的“工具控制语言”角色中,Tcl 有许许多多种内部文件测试和操作功能。每条命令都以 file 开始,正如 file exists name 中一样。其它测试命令(它们都返回布尔值)包括 executableisdirectoryisfileownedreadablewritable

文件信息和操作(再提醒您一次,所有都是以 file 开始)是通过 atimeattributescopydeletelstatmkdirmtimereadlinkrenamerootnamesizestattype 来完成。请注意,在 Windows 或 Mac 环境中运行一些文件信息命令时,可能会返回未定义的数据,因为例如在那些文件系统中没有表示索引节点和符号(硬)链接数据。

使用 file ... 命令而不使用通过 exec 的本机命令的好处在于,前者会提供一个可移植接口。

整个文档只简单地着重于 Tcl 语言的这一部分。然而,在您“品尝”完左边这个 tclsh 示例中这些命令的味道并对此感到满足之后,然后请继续阅读本教程结尾部分列出的“参考资料”。


 

~/tcltk$ tclsh

% nslookup orbdesigns.com

Server: 192.168.1.3

Address: 192.168.1.3#53



Name: orbdesigns.com

Address: 64.81.69.163



% set d [date]

Sun Mar 25 13:51:59 PST 2001

% puts stdout $d



% set d [exec date]

Sun Mar 25 13:52:19 PST 2001

% puts stdout $d

Sun Mar 25 13:52:19 PST 2001



******



% if [catch {open foo r} Chan] {

puts stdout "Sorry, Dave..."n"

}


% gets $Chan

One

% gets $Chan

Two

% eof $Chan

0

% close $Chan

%



 


Tcl 进程和文件 I/O 第 14 页(共15 页)


exec 命令用于显式地执行外部命令。在 Linux 下,当 Tcl 处于交互方式时,可以直接运行大多数外部命令,如左边示例所示。用 exec 运行时,会将程序的 stdout 输出返回到 Tcl,而不是返回到屏幕,这允许将数据赋值给变量。当程序在后台启动时,立即返回的值是程序的 PID。exec 程序可以充分利用 UNIX 环境中的 I/O 重定向和管道。

其它进程命令有 exit(终止正在运行的 Tcl 脚本)和 pid(返回当前或指定进程的 PID),对于出于各种目的的情况,这非常便利。Tcl 不合并任何本机进程控制命令,但可以将 exec 命令与 PID 数据一起使用来实现许多任务。

文件操纵使用下列命令:openclosegetsputsreadtellseekeofflush。如左边所示,在文件打开命令期间 catch 命令对错误检查是有用的。当在遇到新的一行字符之前需要打印程序输出时,如在用户数据提示符中,使用 flush 来写输出缓冲区。

另外一个功能(在受支持的环境中)是以打开文件的方式打开管道的能力。例如,用 set Channel [open "|sort foobar" r] 打开管道通道后,第一个 gets 的输出将是“Eight”(文件数据“One”到“Ten”的输出在 10 个单独的行上按字母顺序排列)。


 

~/tcltk$ cat input01.txt

1 + 2

4 + 5

7 - 9



~/tcltk$ tclsh

% set InFile [open input01.txt r]

file3

% while {[gets $InFile Op] >= 0} {

set Operation "expr $Op"

set Result [eval $Operation]

puts stdout "$Op = $Result"n"

}


1 + 2 = 3



4 + 5 = 9



7 - 9 = -2



%



 


eval 用于动态脚本 第 15 页(共15 页)


在这个示例中,您可以感到 eval 命令的强大功能。在正常情况下,Tcl 解释器以一遍方式(one-pass)操作:它首先对输入的命令行(可能延伸在几个物理行上)进行解析,并执行任何替代。然后开始执行,除非找到不正常或残缺命令。eval 允许第二遍方式(second pass)(或许更精确地讲,是预通过(pre-pass))。因而,可以先动态构造 Tcl 命令,然后进行解析并执行它。

在左边的列表中,输入文件由三行组成,每行都显示了一种算术运算。调用 tclsh 后,文件以只读方式打开并与 $InFile 变量相关联。while 循环每次将一行读入到 $Op 中。然后,通过预先计划将 expr 映射到 $Op 变量来构造整个 Tcl 命令。然后,扩展,求值,从而分配结果。最后,在 stdout 上显示每步操作和结果。

虽然该样本演示了相对琐细的 eval 应用程序,但从概念上讲,可以根据已知语法的输入文件的输入,很容易地将它扩展为动态文件和/或目录处理,或扩展为对文件类型、许可权、访问时间或任何种类的可测试元素的基本操作。

 

Expect 是什么? 第 1 页(共5 页)


Expect 是 Tcl 和 Tk 语言的扩展。Expect 为使交互式程序的脚本编制自动化,提供了简单而功效强大的接口。另外,Expect 使交互式应用程序嵌入 GUI 变得容易。Expect 的开发与 Tcl/Tk 的出现是同时发生的,两者目前的版本都是 5.32。

Expect 的作者是 Don Libes,他在美国国家标准与技术学会(NIST)工作。Expect 主页驻留在 NIST 服务器上。(然而,Expect 和任何相关的商业或非商业产品显然都未经 NIST 认可。)在下面几屏中,将研究从源代码示例目录中精选出来的一些 Expect 脚本示例,并简要概述它的命令语法。

为什么要学习 Expect?引用 Don 论文中的一句话,“使用 expect,使系统管理任务自动化”(USENIX LISA 会议 1990 年 10 月)“……结果是 UNIX 系统管理员的工具箱里充斥着曾经见到过的一些最差的用户界面。只有完全重新设计才能帮助解决所有这些问题,expect 可用来处理许多这些问题。”

 

#!/usr/local/bin/expect --



# ftp-rfc <rfc-number>

# ftp-rfc -index



# retrieves an rfc (or the index) from uunet



exp_version -exit 5.0



if {$argc!=1} {

send_user "usage: ftp-rfc "[#] "[-index]"n"

exit

}



set file "rfc$argv.Z"



set timeout 60

spawn ftp ftp.uu.net

expect "Name*:"

send "anonymous"r"

expect "Password:"

send "bilbrey@orbdesigns.com"r"

expect "ftp>"

send "binary"r"

expect "ftp>"

send "cd inet/rfc"r"

expect "550*ftp>" exit "250*ftp>"

send "get $file"r"

expect "550*ftp>" exit "200*226*ftp>"

close

wait

send_user ""nuncompressing file - wait..."n"

exec uncompress $file



 


使用 Expect 的 RFC 检索 第 2 页(共5 页)


在介绍 Expect 之前,请先研究左边的示例。它只是对在标准 Expect 源代码分发版的示例目录中的版本略加修改而得到的,所有示例都在这个目录下。让我们看一下代码……

这 个程序使 UUNet 档案文件中 IETF RFC(Request for Comment)文档的 FTP 检索自动化。脚本的第一行调用 Expect 外壳。注意,我已经给出了可执行文件的完整路径名。这是最安全的,因为很难知道任何给定用户的路径环境。该脚本先检查 Expect 版本,然后打印用法消息,除非给出正确的自变量数目。

接下来,设置 timeout 值,在下一行产生的 FTP 会话无法正确连接时,防止 Expect 脚本锁住系统资源。脚本的其余部分大多数是几组 expectsend 命令对。每个 expect 命令等待来自产生程序(在本例中是 ftp)的指定输出,后 send 正确响应。注意,在 cd 和 get 指令之后有两个 ftp 错误代码的陷阱。对于每种情况,错误代码 550 与第一个条件匹配,如果为真,则脚本存在。否则,执行 250 代码(表示成功),expect 进入下一条命令。

接收文档之后,脚本向 ftp 会话发出 close 命令。wait 命令挂起脚本处理,直到 ftp 终止为止。最后,脚本将给用户发送一个消息,对已下载的 RFC(或 rfc 下标)进行解压,然后隐式退出(缺省情况下)。


 

#!../expect -f

# wrapper to make passwd(1) be non-interactive

# username is passed as 1st arg, passwd as 2nd



set password [lindex $argv 1]

spawn passwd [lindex $argv 0]

expect "password:"

send "$password"r"

expect "password:"

send "$password"r"

expect eof



 


Expect 的关键,第 1 部分 第 3 页(共5 页)


Expect(作为语言,‘E’大写)有四个关键命令。第一个是 expect(命令,小写‘e’),如果找到匹配,它搜索模式并执行命令。对于每条 expect 命令,可以有几个组,每个组都是由选项标志、与之匹配的模式以及要执行的命令或命令主体组成。缺省情况下,expect“侦听”SDTOUT 和 STDERR,直到找到匹配或 timeout 期满为止。

缺省情况下,使用 Tcl 的字符串匹配设施来匹配模式,它实现文件名替换,类似于 C 外壳模式匹配。-re 标志调用 regexp 匹配,-ex 表明必须是精确匹配,不带通配符或变量扩展。expect 的其它可选标志包括 -i-nocase,前者表示要监控产生的进程,后者强迫在匹配之前将进程输出变为小写。对于完整的说明,在命令提示符下输入 man expect ,以查看 Expect 的系统手册页面文档。

第二个重要命令是 send,它用于为由 Expect 脚本正在监控的进程生成输入。send 合并选项以发送给指定的产生的过程(-i),缓慢地发送(-s,例如,在串行通信中,为了不使缓冲区溢出)以及其它几个选项。


#!/usr/local/bin/expect 

# Script to enforce a 10 minute break

# every half hour from typing -

# Written for someone (Uwe Hollerbach)

# with Carpal Tunnel Syndrome.

# If you type for more than 20 minutes

# straight, the script rings the bell

# after every character until you take

# a 10 minute break.



# Author: Don Libes, NIST

# Date: Feb 26, '95



spawn $env(SHELL)



# set start and stop times

set start [clock seconds]

set stop [clock seconds]



# typing and break, in seconds

set typing 1200

set notyping 600



interact -nobuffer -re . {

set now [clock seconds]



if {$now-$stop > $notyping} {

set start [clock seconds]

} elseif {$now-$start > $typing} {

send_user ""007"

}

set stop [clock seconds]

}



 


Expect 的关键,第 2 部分 第 4 页(共5 页)


左边是称为 carpal 的脚本,它也是来自源代码 Expect 分发版的另一个示例。

spawn 是用于创建新进程的 Expect 命令。它已经出现在我们使用过的每个示例中。在左边,它把路径拖到缺省外壳可执行文件并产生新实例。在这样做时,spawn 返回一个进程标识(在变量 spawn_id 中设置)。这可以在脚本中保存并设置,这给予了 expect 进程控制能力。

interact 是 Expect 用来打开用户与产生进程之间通信的命令。-nobuffer 标志将与模式匹配的字符直接发送给用户。-re 告诉 interact 将接下来的模式用作标准正规表达式,‘.’是与输入时每个字符匹配的模式。在交互方式中,缺省情况下,Expect 的 STDOUT 和 STDERR 流的重定向也返回给用户。


 

使用 Expect 可以完成哪些任务? 第 5 页(共5 页)


当脚本调用交互式程序时,缺省情况下,Expect 拦截所有输入和输出(STDIN、STDOUT 和 STDERR)。这允许 Expect 搜索与程序输出匹配的模式,并将输入发送到产生的进程,以模拟用户交互。另外,Expect 可以将进程的控制传递给用户(如果这样指示的话),或者根据请求控制。

这些特性不仅使 Expect 对于公共管理任务变得非常有用,而且证实了 Expect 有益于构建测试脚本,以在程序开发期间执行 I/O 验证。

最后,有一个极其有用的程序 autoexpect。它本身是一个 Expect 脚本,autoexpect 监控命令行交互式程序,生成精确复制该交互的 Expect 脚本。现在,虽然通常不需要它,但很容易拿几个 autoexpect 会话的结果,概括 expect 模式,然后将它们剪贴到期望的配置中。已经在多处提到过,学习 Expect 的最佳工具是运行 autoexpect 并使用这些结果。

 

array 命令是一种多用途工具,很象 stringarray exists 命令用于测试变量是否作为数组存在,array get 用于将数组转换成列表格式,array set 用于将列表转换为数组,array names 用于返回下标列表,array size 用于返回对下标进行计数的结果。搜索整个数组有它自己的一组四个命令:array startseacharray anymorearray nextelementarray donesearch

虽然设计之初 Tcl 数组是一维的,但有一个模拟多维结构的好方法。因为下标是任意字符串,所以二维数组可以声明如下:



set i 1 ; set j 10



set array($i,$j) 3.14159

incr $j

set array($i,$j) 2.71828



这些数组键实际上分别只是字符串“1,10”和“1,11”,但对访问数据来说,谁知道这之间的差异?

 

#!/usr/bin/tclsh

#

# Demonstrate procedures and

# global scoping briefly



set PI [expr 2 * asin(1.0)]



proc circum {rad} {

global PI

return [expr 2.0 * $rad * $PI]

}



proc c_area {rad} {

global PI

return [expr $rad * $rad * $PI]

}



set rad 3



puts stdout "Area of circle of"

radius $rad is [c_area $rad],"n"

the circumference is"

[circum $rad]."n"



*********

~/tcltk$ ./protest.tcl

Area of circle of radius 3 is 28.2743338823,

the circumference is 18.8495559215.




posted on 2009-05-06 10:57  starspace  阅读(2245)  评论(0编辑  收藏  举报

导航