《Tcl语言基础教程教学内容.doc》由会员分享,可在线阅读,更多相关《Tcl语言基础教程教学内容.doc(32页珍藏版)》请在淘文阁 - 分享文档赚钱的网站上搜索。
1、Good is good, but better carries it.精益求精,善益求善。Tcl语言基础教程-2013/8/13Tcl语言基础|王建军yspringtechTcl语言参考笔记:Tcl语言参考Tcl(ToolCommandLanguage,即工具命令语言;Tcl念作“踢叩”tickle)是一种易学易用易于扩展的脚本语言,实际上包含了两个部分:一个语言和一个库。首先,Tcl是一种简单的脚本语言,主要使用于发布命令给一些交互程序如文本编辑器、调试器和shell。它有一个简单的语法和很强可扩充性,Tcl可以创建新的过程以增强其内建命令的能力。其次,Tcl是一个库包,可以被嵌入应用程序
2、,Tcl的库包含了一个分析器、用于执行内建命令的例程和可以使你扩充(定义新的过程)的库函数。应用程序可以产生Tcl命令并执行,命令可以由用户产生,也可以从用户接口的一个输入中读取(按钮或菜单等)。Tcl和其他编程语言例如C不同,它是一种解释语言而非编译语言。Tcl程序由一系列Tcl命令组成,在运行时由Tcl解释器解释运行。Tcl有大量的用C/C+编写的扩展用于提供Tcl本身不具有的功能。其中使用最广泛的扩展是TK,TK提供了各种OS平台下的图形用户界面GUI(连强大的Python语言都不单独提供自己的GUI,而是提供接口适配到TK上)。另一个流行的扩展包是Exp-ect,它提供了通过终端自动执
3、行命令的能力,例如passwd,ftp,telnet等命令驱动的外壳。一、Tcl程序基本结构1、基本语法Tcl有类似于shell的语法,一条Tcl的命令串包含了一条或多条命令用换行符或分号来隔开,而每一条命令包含了一个域(field)的集合,域使用空白(空格或TAB)分开的,第一个域是一个命令的名字,其它的是作为参数来传给它。Tcl解释器对一个命令的求值过程分为两部分:分析和执行。在分析阶段,Tcl解释器运用规则识别出命令并把命令分成一个个独立的单词,同时进行必要的置换(substitution);在执行阶段,Tcl解释器会把第一个单词当作命令名,并查看这个命令是否有定义,如果有定义就激活这个
4、命令对应的C/C+过程,并把所有的单词作为参数传递给该命令过程,让命令过程进行处理。Tcl的命令名可以是内建的命令也可以是用户建的新命令,在应用程序中可用函数Tcl_CreateCommand来创建新的命令。所有的参数作为字符串来传递,命令自己会按其所需来解释的参数的。另外关于Tcl的注释有一个特殊的要求:必须出现在Tcl解释器期望命令的第一个字符出现的地方,才被当作注释。例如:seta100#Notacommentsetb101;#thisisacomment第一个例子中#及其后面的内容会被当作set的参数而非注释从而导致参数个数错误;但第二个被解释为注释,因为#前面有一个分号,而分号和换行
5、一样被看作是命令的分隔符。脚步命令:(1)eval命令eval命令是一个用来构造和执行TCL脚本的命令,其语法为:evalargarg.它可以接收一个或多个参数,然后把所有的参数以空格隔开组合到一起成为一个脚本,然后对这个脚本进行求值。(2)source命令source命令读一个文件并把这个文件的内容作为一个脚本进行求值。例如:sourcee:/tcl&c/hello.tcl注意路径的描述应该和UNIX相同,使用/而不是。2、置换(Substitution)setx10sety100+x上面命令执行后,y的值是“100+x”而不是我们期望的110。这是因为Tcl解释器在分析命令时,把所有的命令
6、参数都当作字符串看待,所以x被看作了字符串“100+x”的一部分。如果我们想使用x的值10,就必须告诉Tcl解释器:我们在这里期望的是变量x的值,而非字符x。怎么告诉Tcl解释器呢,这就要用到Tcl语言中提供的置换功能。Tcl提供三种形式的置换:变量置换、命令置换和反斜杠置换。每种置换都会导致一个或多个单词本身被其他的值所代替。置换可以发生在包括命令名在内的每一个单词中,而且置换可以嵌套。(1)变量置换(variablesubstitution)变量置换由一个符号标记,变量置换会导致变量的值而非变量(标识符)本身被插入到字符串中。setx10sety100+$x这时,y的值还不是我们想要的值1
7、10,而是10+100,因为Tcl解释器把10+100看成是一个字符串而不是表达式;y要想得到值110,还必须用命令置换,使得Tcl会把10+100看成一个表达式并求值。(2)命令置换(commandsubstitution)命令置换是由括起来的Tcl命令及其参数,命令置换会导致某一个命令的所有或部分单词(参数)被另一个命令的结果所代替。setx10setyexpr100+$x这时,y的值就是110了。这里当Tcl解释器遇到字符时,它就会把随后的expr作为一个命令名,从而激活与expr对应的C/C+过程,并把expr命令中变量置换后得到的10+110传递给该命令过程进行处理。注意,中必须是一
8、个合法的Tcl脚本,长度不限。中脚本的值为最后一个命令的返回值,例如:有了命令置换,实际上就表示命令之间是可以嵌套的,即一个命令的结果可以作为别的命令的参数。(3)反斜杠置换(backslashsubtitution)Tcl语言中的反斜杠置换类似于C语言中反斜杠的用法,主要用于在单词符号中插入诸如换行符、空格、$等被Tcl解释器当作特殊符号对待的字符。例如:setmsgmultiplespace如果没有的话,Tcl会报错,因为解释器会把这里最后两个单词之间的空格认为是分隔符,于是发现set命令有多于两个参数,从而报错。加入了后,空格不被当作分隔符,multiplespace被认为是一个单词(w
9、ord)。(4)双引号和花括号除了使用反斜杠外,Tcl提供另外两种方法来使得解释器把分隔符和置换符等特殊字符当作普通字符,而不作特殊处理,这就要使用双引号和花括号()。Tcl解释器对双引号中的各种分隔符将不作处理,但是对换行符及和两种置换符会照常处理。而在花括号中,所有特殊字符都将成为普通字符,失去其特殊意义,Tcl解释器不会对其作特殊处理。二、Tcl变量数据类型Tcl只支持一种数据类型:字符串(string)。所有的命令,命令的所有的参数,命令的结果,所有的变量都是字符串。请牢记这一点,所有的东西都是字符串。(然而字符串的实际解释是依赖于其所处上下文。它有三种形式:命令,表达式和表)。Tcl
10、变量根据其结构的复杂程度分为“简单变量”和“数组变量”两类。(1)简单变量一个Tcl的简单变量包含两个部分:名字和值,其中名字和值都可以是任意字符串。不过为了更好的使用变量置换功能,变量名最好按CC+语言中标识符的命名规则命名。这是因为Tcl解释器在分析一个变量置换时,只把从符号往后直到第一个不是字母、数字或下划线的字符之间的单词符号作为要被置换的变量的名字。例如:seta2seta.15setb$a.1在最后一个命令行,我们希望把变量a.1的值付给b,但是Tcl解释器在分析时只把$符号之后直到第一个不是字母、数字或下划线的字符(这里是.)之间的单词符号(这里是a)当作要被置换的变量的名字,所
11、以Tcl解释器把a置换成2,然后把字符串“2.1”付给变量b。这显然与我们的初衷不同。当然,如果变量名中有不是字母、数字或下划线的字符又要用变量置换,可以用花括号把变量名括起来。例如:setb$a.1(2)数组变量数组是一些元素的集合。Tcl的数组和一般编程语言中的数组有很大的区别。在Tcl中,不能单独声明一个数组,数组只能和数组元素一起声明。数组中数组元素的名字包含两部分:数组名和数组中元素的名字,Tcl中数组元素的名字(下标可以为任何字符串。例如:setday(monday)1setday(tuesday)2(3)重用结构及其操作1、string结构及其操作因为TCL把所有的输入都当作字符
12、串看待,所以TCL提供了较强的字符串操作功能【format】语法:formatformatstringvluevalue.format命令类似于ANSIC中的sprintf函数,它按formatstring提供的格式,把各个value的值组合到formatstring中形成一个新字符串,并返回。例如:setmsgformat%sis%dyearsold$name$age【scab】语法:scanstringformatstingvarNamevarName.scan命令可以认为是format命令的逆,其功能类似于ANSIC中的sscanf函数。它按formatsting提供的格式分析strin
13、g字符串,然后把结果存到变量varName中,注意除了空格和TAB键之外,string和formatsting中的字符和%必须匹配。例如:scansome2634some%d%dab【regexp】语法:regexpswitchs-expstringmatchVarsubMatchVarsubMatchVar.regexp命令用于判断正规表达式exp是否全部或部分匹配字符串string,匹配返回1,否则0。regexp可以设置一些开关(switchs,来控制匹配的具体方式,如:-nocase,-line等,其中-表示这后面再没有开关(switchs了,即使后面有以-开头的参数也被当作正规表达式
14、的一部分。如果regexp命令后面有参数matchVar和subMatchVar,则所有的参数被当作变量名,如果变量不存在,就会被生成。regexp把匹配整个正规表达式的子字符串赋给第一个变量,匹配正规表达式的最左边的子表达式的子字符串赋给第二个变量,依次类推,例如:regexp(0-9+)*(a-z+)thereis100applestotalnumword1puts$total,$num,$word显示结果为:100apples,100,apples其他具体的正则表达式规则请参考Tcl语言手册【regsub】语法:regsubswitchsexpstringsubSpecvarnamere
15、gsub的第一个参数是一个整个表达式,第二个参数是一个输入字符串,这一点和regexp命令完全一样,也是当匹配时返回1,否则返回0。不过regsub用第三个参数的值来替换字符串string中和正规表达式匹配的部分,第四个参数被认为是一个变量,替换后的字符串存入这个变量中。例如:regsubthereTheylivetherelivestheirxputs$x显示结果为:Theylivetheirlives【sting】语法:stringsubcmdargarg.string命令具有强大的操作字符串的功能,并通过其中的subcmd来区别具体将要执行的string操作。他们中的一些如下:strin
16、glengthstringstringrangestringfirstlaststringindexstringcharIndexstringtolowerstringfirstlaststringreplacestringfirstlastnewstringstringequal-nocase-lengthintstring1string2stringmatch-nocasepatternstring如果pattern匹配string,那么返回1,否则返回0.stringcompare-nocase-lengthintstring1string2如果有-length参数,那么只比较前int个
17、字符stringfirststring1string2startindex在string2中从头查找与string1匹配的字符序列,如果找到,那么就返回匹配的第一个字母所在的位置(0-based)。如果没有找到,那么返回-1。stringtrimstringchars返回值为:从string字符串的首尾删除掉了字符集合chars中的字符后的字符串。如果没有给出chars,那么将删除掉spaces、tabs、newlines、carriagereturns这些字符.2、list结构及其操作list这个概念在TCL中是用来表示集合的。TCL中list是由一堆元素组成的有序集合,list可以嵌套定义
18、,list每个元素可以是任意字符串,也可以是list。Tcl提供了很多基本命令对list进行操作:【list】语法:listvaluevalue.这个命令生成一个list,list的元素就是所有的value【concat】语法:concatlistlist.这个命令把多个list合成一个list.【llength】语法:llengthlist返回list的元素个数。【lindex】语法:lindexlistindex返回list的第index个(0-based)元素。【lrange】语法:lrangelistfirstlast返回list的第first(0-based)到第last(0-bas
19、ed)元素组成的串,如果last的值是end。就是从第first个直到串的最后。【linsert】语法:linsertlistindexvaluevalue.?返回一个新串,新串是把所有的value参数值插入list的第index个(0-based)元素之前得到。【lappend】语法:lappendvarnamevaluevalue.?把每个value的值作为一个元素附加到变量varname后面,并返回变量的新值,如果varname不存在,就生成这个变量。【lreplace】语法:lreplacelistfirstlastvaluevalue.返回一个新串,新串是把list的第firs(0-
20、based)t到第last个(0-based)元素用所有的value参数替换得到的。如果没有value参数,就表示删除第first到第last个元素。【lsort】语法:lsortoptions?list这个命令返回把list排序后的串。options可以是如下值:-ascii按ASCII字符的顺序排序比较.这是缺省情况。-dictionary按字典排序,与-ascii不同的地方是:(1)不考虑大小写(2)如果元素中有数字的话,数字被当作整数来排序.【lsearch】语法:lsearch-optionlistpattern返回list中第一个匹配模式pattern的元素的索引,如果找不到匹配就
21、返回-1。option取-exact、-glob、-regexp是三种模式匹配的技术,缺省时使用-glob匹配。【join】语法:joinlistjoinString这个命令把list的所有元素合并到一个字符串中,中间以joinString分开。缺省的joinString是空格。【split】语法:splitstringsplitChars把字符串string按分隔符splitChars分成一个个单词,返回由这些单词组成的串。如果splitChars是一个空字符,string被按字符分开。如果splitChars没有给出,以空格为分隔符。3、file结构及其操作TCL提供了丰富的文件操作的命令
22、。通过这些命令你可以对文件名进行操作(查找匹配某一模式的文件)、以顺序或随机方式读写文件、检索系统保留的文件信息(如最后访问时间)。3.1基本文件I/Opwd和UNIX下的pwd命令完全一样,没有参数,返回当前目录的完整路径。cd命令也和UNIX命令也一样,使用一个参数,可以把工作目录改变为参数提供的目录。以下这个名为tgrep的过程,可以说明TCL文件I/O的基本特点:proctgreppatternfilenamesetfopen$filenamerwhilegets$flineifregexp$pattern$lineputsstdout$lineclose$fopen命令返回一个字符串
23、用于表识打开的文件。当调用别的命令(如:gets,puts,close,对打开的文件进行操作时,就可以使用这个文件标识符。TCL有三个特定的文件标识:stdin,stdout和stderr,分别对应标准输入、标准输出和错误通道,任何时候你都可以使用这三个文件标识。3.2随即文件访问默认文件输入输出方式是连续的:即每个gets或read命令返回的是上次gets或read访问位置后面的字节,每个puts命令写数据是接着上次puts写的位置接着写。TCL提供了seek,tell和eof等命令使用户可以非连续访问文件。每个打开的打开文件都有访问点,即下次读写开始的位置。文件打开时,访问点总是被设置为文
24、件的开头或结尾,这取决于打开文件时使用的访问模式。每次读写后访问位置按访问的字节数后移相应的位数。seekfileIdoffsetorigin把fileId标识的文件的访问点设置为相对于origin偏移量为offset的位置。origin可以是start,current,end,默认是start。tellfileId返回fileId标识的文件的当前访问位置。eoffileId如果到达fileId标识的文件的末尾返回1,否则返回0。3.3文件操作和获取文件信息【grob】语法:globswitchespatternpattern.glob命令的模式采用stringmatch命令的匹配规则,返回匹
25、配这个(些)模式的所有文件的列表。如:glob*.c*.hglob*/只返回当前目录的所有子目录。【file】语法:filesubcmdargarg.file通过众多的subcmd子操作提供强大的文件操作和访问能力。他们中的一些如下:fileexistsnamefileextensionnamefilemkdirdirdir.filecopy-force-sourcetargetfilerename-force-sourcetargetfiledelete-force-pathnamepathname.三、Tcl表达式(1)操作数Tcl表达式的操作数通常是整数或实数。整数一般是十进制的,但如果
26、整数前两个字符是0x则这个整数被看作是十六进制的。Tcl的实数的写法与ANSIC中完全一样。(2)运算符Tcl中的运算符的语法形式和用法跟ANSIC中完全一致。另外,Tcl中提供了大量的常用数学函数来增强自身的数学运算和处理能力,这些函数的形式和用法大都与ANSIC中的数学函数完全一致。四、Tcl流程控制TCL中的控制流和C语言类似,包括if、switch、for、while、foreach、break、continue等命令。(1)条件语句if$x0.elseif$x1.elseif$x2.else.注意,上例中一定要写在上一行,因为如果不这样,TCL解释器会认为if命令在换行符处已结束,下
27、一行会被当成新的命令,从而导致错误的结果。在Tcl的其他流程控制命令的书写中也要注意这个问题。书写中还要注意的一个问题是if和之间应该有一个空格,否则TCL解释器会把if作为一个整体当作一个命令名,从而导致错误。switchoption$xa-bincrt1cincrt2defaultincrt3其中可选参数option,表示进行匹配的方式。TCL支持三种匹配方式:-exact方式,-glob方式,-regexp方式,缺省情况表示-glob方式。-exact方式表示的是精确匹配,-glob方式的匹配方式和stringmatch命令的匹配方式相同,-regexp方式是正规表达式匹配方式。条件中a
28、的后面跟一个表示使用和下一个模式相同的脚本,default表示匹配任意值。一旦switch命令找到一个模式匹配,就执行相应的脚本,并返回脚本的值作为switch命令的返回值。(2)循环语句forinittestreinitbody参数init是一个初始化脚本,第二个参数test是一个表达式,用来决定循环什么时候中断,第三个参数reinit是一个重新初始化的脚本,第四个参数body也是脚本。如:setbforsetiexprllength$a-1$i=0incri-1lappendblindex$a$iwhile的例子假设变量a是一个链表,下面的脚本把a的值复制到b:setbsetiexprll
29、ength$a-1while$i=0lappendblindex$a$iincri-1foreach命令的两种形式:1、foreachvarNamelistbody第一个参数varName是一个变量,第二个参数list是一个表(有序集合),第三个参数body是循环体。每次取得链表的一个元素,都会执行循环体一次。如:setbforeachi$asetblinsert$b0$i2、foreachvarlist1list1varlist2list2.body本语句是上一个的增强形式:可以有多个变量、列表对参与条件判断五、Tcl过程TCL支持过程的定义和调用,在TCL中,过程可以看作是用TCL脚本实现
30、的命令,效果与TCL的固有命令相似。我们可以在任何时候使用proc命令定义自己的过程,TCL中的过程类似于C中的函数。TCL中过程是由proc命令产生的。例如:procaddxyexpr$x+$yproc命令的第一个参数是你要定义的过程的名字,第二个参数是过程的参数列表,参数之间用空格隔开,第三个参数是一个TCL脚本,代表过程体。proc生成一个新的命令,可以象固有命令一样调用:add123在定义过程时,你可以利用return命令在任何地方返回你想要的值。return命令迅速中断过程,并把它的参数作为过程的结果。(1)局部变量和全局变量对于在过程中定义的变量,因为它们只能在过程中被访问,并且当
31、过程退出时会被自动删除,所以称为局部变量;在所有过程之外定义的变量我们称之为全局变量。TCL中局部变量和全局变量可以同名,两者的作用域的交集为空:局部变量的作用域是它所在的过程的内部;全局变量的作用域则不包括任何过程的内部。这一点和C语言有很大的不同。如果我们想在过程内部引用一个全局变量的值,可以使用global命令。(2)缺省参数和可变参数可以定义具有缺省参数值的过程,我们可以为过程的部分或全部参数提供缺省值,如果调用过程时未提供那些参数的值,那么过程会自动使用缺省值赋给相应的参数。和CC+中具有缺省参数值的函数一样,有缺省值的参数只能位于参数列表的后部,即在第一个具有缺省值的参数后面的所有
32、参数,都只能是具有缺省值的参数。TCL的过程定义还支持可变个数的参数,如果过程的最后一个参数是args,那么就表示这个过程支持可变个数的参数,位于args以前的参数象普通参数一样处理,但任何附加的参数都需要在过程体中作特殊处理,过程的局部变量args将会被设置为一个列表,其元素就是所有附加的变量。如果没有附加的变量,args就设置成一个空串。(3)引用:upvar命令语法:upvarlevelotherVarmyVarotherVarmyVar.upvar命令使得用户可以在过程中对全局变量或其他过程中的局部变量进行访问。upvar命令的第一个参数otherVar是我们希望以引用方式访问的参数的
33、名字,第二个参数myVar是这个过程中的局部变量的名字,一旦使用了upvar命令把otherVar和myVar绑定,那么在过程中对局部变量myVar的读写就相当于对这个过程的调用者中otherVar所代表的局部变量的读写。下面是一个例子:proctempargupvar$argbsetbexpr$b+2procmyexpvarseta4tempareturnexpr$var+$a则:myexp713这个例子中,upvar把$arg(实际上是过程myexp中的变量a)和过程temp中的变量b绑定,对b的读写就相当于对a的读写。upvar命令语法中的level参数表示:调用upvar命令的过程相对于我们希望引用的变量myVar在调用栈中相对位置。例如:upvar2otherx这个命令使得当前过程的调用者的调用者中的变量other,可以在当前过程中利用x访问。缺省情况下,level的值为1,即当前过程(上例中的temp)的调用者(上例中的myexp)中的变量(上例中myexp的a)可以在当前过程中利用局部变量(上例中temp的b)访问。如果要访问全局变量可以这样写:upvar0otherx那么,不管当前过程处于调用栈中的什么位置,都可以在当前过程中利用x访问全局变量other。-
限制150内