tcl编程

0. 基础语法

0.1 普通变量

#变量赋值
set x "This is a string"
set y 1.24

#打印变量
puts $x
puts $y

0.2 list, 列表

#列表赋值
set la [list 'a' 'b' {'c' 'd'}]

#llength: 获取list长度
puts [llength $la] ; # 3

#lindex: 根据idx获取元素
set e2 [lindex $la 2]
puts $e2 ; # 'c' 'd'

#lsearch: 在list中查找指定元素
lsearch $la 'b'; # 1, 返回'b'的idx.

#lappend: 追加元素, 可以追加多个
lappend la 'ef' 'gh'; # 注意会修改la本身
puts $la; # 'a' 'b' {'c' 'd'} 'ef' 'gh'

#linsert: 在指定位置插入元素, 可以插入多个
set la_new [linsert $la 1 'x' 'y' 'z']
puts $la_new ; # 'a' 'x' 'y' 'z' 'b' {'c' 'd'}

#lrange: 返回first~last的元素


#lreplace: 替换first~last的元素

#lsort: list排序

#concat: 多个list合并为一个list
set la [list 'a' 'b']
set lb [list 'x' 'y']
set lc [concat la lb]
puts $lc ; # 'a' 'b' 'x' 'y'

0.3 array, 数组

array set arr {}                 ; # 创建空array, 清空已有array

array set arr {a 1 b 2 c 3 de 4} ; # array设置元素, 设置多个
set arr(f) 5                     ; # array设置元素, 设置一个

puts [array size arr]            ; # 5, array长度, key-val对的数目

puts $arr(de)                    ; # 4, 获取key de对应的值, 注意, 不需要在de外面加引号. 

#遍历array
#   de -> 4
#   a -> 1
#   f -> 5
#   b -> 2
#   c -> 3
foreach k [array names arr] {
    set v $arr($k)
    puts "$k -> $v"
}

#判断arr是否存在:
if [array exists arr] {
    puts "array arr exists."
}

#判断key是否存在: info exists arr($k)
#   $arr(b) = 2
if [info exists arr(b)] {
    puts "\$arr(b) = $arr(b)"
}

0.4 循环

0.4.1 for

set la [list 'a' 'b' 'c' 'd' 'e']
for {set i 0} {$i<[llength $la]} {incr i} {
    puts "$i: la[$i]"
}

0.4.2 foreach

#待循环的list
set la [list 'a' 'b' 'c' 'd' 'e']
set lb [list '1' '2' '3' '4' '5' '6']

# 循环单个list, 每次取一个元素
foreach ele $la {
    puts "$ele"
}

# 循环单个list, 每次取多个元素
foreach {ele0 ele1} $la {
    puts "$ele0, $ele1"
}

# 循环多个list, 循环次数以最长的list为准, 不足的补空元素
foreach ea $la eb $lb {
    puts "$ea: $eb"
}

0.5 分支

0.5.1 if..else

set a 100
if {$a == 10} {
    puts "\$a is 10"
} elseif {$a == 20} {
    puts "\$a is 20"
} else {
    puts "\$a is neither 10 nor 20"
}

0.5.2 switch

proc get_instr {location name} {
    set value 0x00
    switch $location {
        "SMSG" {
            switch $name {
                "BYPASS"    {set value 0x00}
                "FLOAD_RUN" {set value 0x30}
                default     {puts "Error: cannot find instr for $location.$name"}
            }
        }
        "PROC" {
            switch $name {
                "BYPASS"   {set value 0x00}
                "BIST_RUN" {set value 0x11}
                default    {puts "Error: cannot find instr for $location.$name"}
            }
        }
        default {puts "Error: first param must be SMSG or PROC"}
    }
    return value
}

puts [get_instr "PROC" "BIST_RUN"]

1. 从命令行获取参数(好像并不是很强大)

array set a0 $argv
foreach opt {-i -o} {
    puts "$opt -> $a0($opt)"
}

运行

$ ex.tcl -i abc -o xyz
-i -> abc
-o -> xyz

2. 给proc定义参数(部分命令只在dc或pt中有效)

proc display {args} {
    parse_proc_arguments -args $args my_arg
    return [expr $my_arg(-a) + $my_arg(-b)]
}

define_proc_attribute display -info "process two number" -define_args {
    {-a "the 1st number" a string required}
    {-b "the 2nd number" b string required}
}

display -a 5 -b 6

3. 读写文件

3.1 按行读入文件

set FH [open ../rpt/$file r]
while {[gets $FH line] >= 0} {
    puts $line
}
close $FH

3.2 写文件

set FH [open "file.txt" w]
puts -nonewline $FH "hello"
puts $FH " world."
close $FH

4. 正则匹配

if [regexp {^(\w+)} $line all_match match1] {
    puts $match1
}

5. catch

当运行的命令出现错误时, catch语句返回1

if [catch {open file.txt r} FH] {
    puts "Error: file.txt not opened"
} else {
    puts "open file.txt"
    while {[gets $FH line] >= 0} {
        puts $line
    }
}

6. 非整数运算

set a [expr 100 * double(1)/3] # 返回33.333333333333336%
                               # 如果不使用double, 则只会返回整数部分
                               # [expr 100 * 1/3] 返回33
                               # [expr 1/3 * 100] 返回0
puts "$a%" # 33.333333333333336%

set b [format "%.2f" $a]
puts "$b%" # 33.33% 

7. 四舍五入, 伪随机数

# 四舍五入
set a 2.3
set b [expr round($a)] #注意, 要通过expr调用, 调用时后面要加圆括号
puts $b # 2, 四舍五入2.3变为2

# 生成0~1之间的伪随机小数
set c [expr rand()] # 注意, 要通过expr调用.
puts $c # 0.45033995969519

# 生成0~9之间的伪随机整数
set d [expr round(9*rand())]
puts $d # 5

8. eval和exec

# eval 动态执行tcl语句
set cmd "puts 123"
eval $cmd # 123

# exec 执行unix shell命令(或windows cmd命令)
set a [exec find . -name "run*"]
puts $a # 打印出find命令的标准输出

9. 进制转换

binary用于操作二进制字符串.
包含四个子命令:

binary format: 将普通tcl字符串转换为二进制字符串.
binary scan  : 将二进制字符串转为普通tcl字符串.
binary encode: 将二进制字符串进行编码.
binary decode: 将编码后的二进制字符串进行解码.

二进制字符串的范围是\u0000~\u00FF, 所以如果不想丢失数据, 需要先进行转换.

# 普通tcl字符串, 内容是16进制的字符
set hex_str FF00CC

# 将普通tcl字符串$hex转为二进制格式,
#    H : 表示待处理的字符串中是16进制字符
#    * : 表示待处理的字符串list长度任意??
set bin_fmt [binary format H* $hex_str]

# 将二进制数据($bin_fmt)转为普通tcl字符串, 存储到变量$bits中
binary scan $bin_fmt B* bits

# 打印$bits
puts $bits ;# 1111111110000000011001100


# 将$bits格式化为特定长度的str, %0*s, 中的*, 表示通过变量指定长度
set width 28
set bits_28 [format "%0*s" $width, $bits]
puts $bits_28 ; 00001111111110000000011001100

#遍历$bits_28
set len [string length $bits_28]
for {set i 0} {$i<=[expr $len-1]} {incr i} {
    set i_str [expr $len-1-$i]
    puts "$i -> [string index $bits_28 $i_str]"
}

# 0 -> 0
# 1 -> 0
# 2 -> 1
# 3 -> 1
# 4 -> 0
# 5 -> 0
# 6 -> 1
# 7 -> 1
# 8 -> 0
# 9 -> 0
# 10 -> 0
# 11 -> 0
# 12 -> 0
# 13 -> 0
# 14 -> 0
# 15 -> 0
# 16 -> 1
# 17 -> 1
# 18 -> 1
# 19 -> 1
# 20 -> 1
# 21 -> 1
# 22 -> 1
# 23 -> 1
# 24 -> 0
# 25 -> 0
# 26 -> 0
# 27 -> 0

posted @ 2022-08-02 14:53  编程驴子  阅读(553)  评论(0编辑  收藏  举报