|
楼主 |
发表于 2020-4-14 16:24
|
显示全部楼层
TCL语法简介8 e6 J6 g9 J1 v; \
6 S6 e& O0 l; A0 b1.基本语法- D5 \' Y& I+ D2 i- T6 v: i$ E: v2 K
TCL语言的语法实际上是一些TCL解释器怎样对TCL命令进行分析的规则的集合,TCL 的基本语法类似于 UNIX shell:命令由用空格或 TAB 分隔的一个或多个字段组成。第一个字段是命令的名字,它可以是内置命令、特定于应用的命令、或者是由一系列的 TCL命令组成的过程。在第一个单词后面的字段都作为参数传递给命令。
3 W" e+ P, m& T$ Z. m' e(1)注释
. f$ H* X2 ?* g) o J1 z0 h关于TCL的注释有一个特殊的要求:'#'必须出现在TCL解释器期望命令的第一个字符出现的地方,才被当作注释。8 X4 h7 l1 `0 g1 x" t
例如:
* G4 R. O B8 q0 F4 Pset a 1 # Not a comment
1 z% O% q$ C: P6 Kset b 2 ; # this is a comment- x4 K0 b7 |0 m* H5 G) B
第一个例子中'#'及其后面的内容会被当作set 的参数而非注释从而导致参数个数错误;但第二个被解释为注释,因为'#'前面有一个分号,而分号和换行一样被看作是命令的分隔符。7 J% r; t0 a$ s7 @6 Y0 {% F) l
(2)eval命令
0 M$ |4 E, R1 D2 y9 F. B2 Meval命令是一个用来构造和执行TCL脚本的命令,其语法为:
4 g5 Y+ G% Q3 z; D) h& Y+ S7 S; s+ _. ^% v$ y7 f
" j- u' n0 P2 t R5 j0 C
eval arg arg ...! E5 E+ h, {' }2 [( U
它可以接收一个或多个参数,然后把所有的参数以空格隔开组合到一起成为一个脚本,然后对这个脚本进行求值。* C, l! U* [8 K4 I4 @
(3)source命令) L9 o* [2 Z$ y& R) F7 x' W
source命令读一个文件并把这个文件的内容作为一个脚本进行求值。例如:
s3 F& r% w( R/ p8 z$ s9 [0 Hsource e:/tcl&c/hello.tcl
2 G/ t j# i( H% F, d2 s7 K" Z注意路径的描述应该和UNIX相同,使用'/'而不是'\'。" c4 g: I' O9 J9 L: m( k
2.置换(Substitution)
R9 ^* g8 T( M* i+ j2 wset x 107 r1 k$ I8 I1 I& l
set y 100 + x1 G! S' P, u/ ?5 _! @) w2 h; j
上面命令执行后,y的值是“100 + x”而不是我们期望的110。这是因为TCL解释器在分析命令时,把所有的命令参数都当作字符串看待,所以x 被看作了字符串“100 + x”的一部分。如果我们想使用x的值'10' ,就必须告诉TCL解释器:我们在这里期望的是变量x的值,而非字符'x'。怎么告诉TCL解释器呢,这就要用到TCL语言中提供的置换功能。9 S \5 m. j* w" H
% e2 n) v, b R3 e _% T3 }
# m( d; p6 p* k% v& L; b) _) q
TCL提供三种形式的置换:变量置换、命令置换和反斜杠置换。每种置换都会导致一个或多个单词本身被其他的值所代替。置换可以发生在包括命令名在内的每一个单词中,而且置换可以嵌套。3 W5 U+ P' J$ S% u- C) x; `2 i! C
(1)变量置换(variable substitution)
' g4 X3 |% _* ]变量置换由一个$符号标记,变量置换会导致变量的值而非变量(标识符)本身被插入到字符串中。
7 S; S1 T* B. V! D9 ]set x 10/ R4 ^9 A! f* k* D& c w
set y 100 + $x
' O4 Q2 L. J8 Q& ]& ?
* v- z3 b9 }3 A) p5 R4 `这时,y的值还不是我们想要的值110,而是10+100,因为TCL解释器把10+100看成是一个字符串而不是表达式;y要想得到值110,还必须用命令置换,使得TCL会把10+100看成一个表达式并求值。
, Q; B5 _$ y6 p8 ^3 w- e5 `(2)命令置换(command substitution)! f% l! t! j/ Y- x% p
命令置换是由[ ]括起来的TCL命令及其参数,命令置换会导致某一个命令的所有或部分单词(参数)被另一个命令的结果所代替。
9 m$ ?# t2 [0 d9 e9 Zset x 102 X1 [8 v" i) g$ Y$ b+ b' B
set y [expr 100+$x]
2 m {1 `8 v, {7 Y' P
# s) w& E+ d9 c8 M7 S, }这时,y的值就是110了。这里当TCL解释器遇到字符'['时,它就会把随后的expr作为一个命令名,从而激活与expr对应的C/C++过程,并把expr命令中变量置换后得到的'10+110'传递给该命令过程进行处理。
+ A' _$ {2 I$ ~( |6 P注意,[ ]中必须是一个合法的TCL脚本,长度不限。[ ]中脚本的值为最后一个命令的返回值。 ; X3 G% R% _) X( o- l
有了命令置换,实际上就表示命令之间是可以嵌套的,即一个命令的结果可以作为别的命令的参数。$ p0 T# H1 h2 Q7 w4 s$ t; j
(3)反斜杠置换(backslash subtitution)
# i5 S6 |+ n, y5 a. W0 ITCL语言中的反斜杠置换类似于C语言中反斜杠的用法,主要用于在单词符号中插入诸如换行符、空格、[、$等被TCL解释器当作特殊符号对待的字符。例如:# E7 M# p! M3 F+ v2 Q1 @
9 F5 }# D& e% R- E4 k: |% O/ A: {3 D% d1 m
) p# W/ K" g/ o) \% {; i! M+ W
set msg multiple\ space 7 e: t# a2 E7 c
如果没有'\'的话,TCL会报错,因为解释器会把这里最后两个单词之间的空格认为是分隔符,于是发现set命令有多于两个参数,从而报错。加入了'\'后,空格不被当作分隔符,'multiple space'被认为是一个单词(word)。+ V1 H. A2 T3 \# a% z
(4)双引号和花括号2 \. k; V9 ?6 c
除了使用反斜杠外,TCL提供另外两种方法来使得解释器把分隔符和置换符等特殊字符当作普通字符,而不作特殊处理,这就要使用双引号“”和花括号{}。TCL解释器对双引号中的各种分隔符将不作处理,但是对换行符及$和[]两种置换符会照常处理。而在花括号中,所有特殊字符都将成为普通字符,失去其特殊意义,TCL解释器不会对其作特殊处理。
3 ^ M6 t0 M K8 U7 r6 j) ?
% L9 I- d! @; F& p) V
6 v1 s; R' a$ V+ j l/ { }" F7 ^8 ?+ l9 w: {! I
# M5 b" j* ^* n3 K2 _/ Y% x0 p
; @6 R) C9 n- ~ O, S5 n
|
|