如何在TCL中创建和迭代哈希散列?
如果我有以下数据:
foo = {
a => {
aa => { aa1 aa2 aa3 }
ab => { ab1 ab2 ab3 }
ac => { ac1 ac2 ac3 }
}
b => {
ba => { ba1 ba2 ba3 }
bb => { bb1 bb2 bb3 }
bc => { bc1 bc2 bc3 }
}
c => {
ca => { ca1 ca2 ca3 }
cb => { cb1 cb2 cb3 }
cc => { cc1 cc2 cc3 }
}
}
如何通过一次插入一个叶节点数据项来创建此类哈希。类似的东西:
lappend foo(a)(ab)“ab1”
那么如何迭代所有数据元素?像:
foreach key in foo {
foreach sub_key in foo($key) {
foreach elem in foo($key)($sub_key) {
puts "foo\($key\)\($sub_key\) is $elem"
}
}
}
编辑: 不幸的是,我无法访问更新的'dict'结构。
答案 0 :(得分:8)
假设你正在使用Tcl 8.5+,那么词典就是你的选择:
简单地完成字典定义:
set foo {
a {
aa { aa1 aa2 aa3 }
ab { ab1 ab2 ab3 }
ac { ac1 ac2 ac3 }
}
b {
ba { ba1 ba2 ba3 }
bb { bb1 bb2 bb3 }
bc { bc1 bc2 bc3 }
}
c {
ca { ca1 ca2 ca3 }
cb { cb1 cb2 cb3 }
cc { cc1 cc2 cc3 }
}
}
或以编程方式定义:
set foo [dict create]
foreach first {a b c} {
dict update foo $first subdict {
foreach second {a b c} {
foreach third {1 2 3} {
dict lappend subdict "$first$second" "$first$second$third"
}
}
}
}
并输出:
dict for {key1 subdict} $foo {
dict for {key2 list} $subdict {
foreach elem $list {
puts "$key1\t$key2\t$elem"
}
}
}
编辑:将数组解决方案(非dict)移到单独的答案中。
答案 1 :(得分:7)
如果您没有使用Tcl 8.5,那么您可以使用数组。请注意,数组是一维的,但键是一个可用于伪造多维度的任意字符串:
array set foo {}
foreach first {a b c} {
foreach second {a b c} {
foreach third {1 2 3} {
lappend foo($first,$first$second) "$first$second$third"
}
}
}
parray data
并输出 - 注意:与字典键不同,数组键是无序的:
foreach key [array names foo] {
foreach elem $foo($key) {
puts "$key\t$elem"
}
}
如果给你钥匙(例如'b'和'bc'),你可以得到这样的价值:
set key1 b
set key2 bc
foreach elem $foo($key1,$key2) {puts $elem}
答案 2 :(得分:1)
如果你只是想在没有dict命令的情况下遍历一个dict(它只是一个键值对列表),那么你可以简单地使用foreach的超棒:
set foo {
a {
aa { aa1 aa2 aa3 }
ab { ab1 ab2 ab3 }
ac { ac1 ac2 ac3 }
}
b {
ba { ba1 ba2 ba3 }
bb { bb1 bb2 bb3 }
bc { bc1 bc2 bc3 }
}
c {
ca { ca1 ca2 ca3 }
cb { cb1 cb2 cb3 }
cc { cc1 cc2 cc3 }
}
}
foreach {key value} $foo {
foreach {sub_key sub_value} $value {
foreach elem $sub_value {
puts "foo\($key\)\($sub_key\) is $elem"
}
}
}
另一方面,如果没有dict命令,一次插入一个元素会很痛苦:
set foo {}
lappend foo a {}
set a_index [lsearch $foo a]
set a_value_index [expr {$a_index+1}]
set a_value [lindex $foo $a_value_index]
lappend a_value aa {}
lset foo $a_value_index $a_value
# it is now too painful for me to continue :-(
幸运的是,您可以使用dict命令的纯tcl实现:forward-compatible dict
答案 3 :(得分:1)
如果您没有Tcl 8.5字典,请使用键控列表命令完成工作。您可以谷歌搜索其中一个术语:keylget,keylset。
package require Tclx
# Create the nested structure
catch {unset foo}
foreach key1 {a b c} {
foreach key2 {a b c} {
catch {unset element}
foreach key3 {1 2 3} {
lappend element "$key1$key2$key3"
}
keylset foo $key1.$key1$key2 $element
}
}
# Access the nested structure
foreach key1 {a b c} {
foreach key2 {a b c} {
set elementList [keylget foo $key1.$key1$key2]
foreach element $elementList {
puts "foo\\$key1\\$key1$key2\\$key3 = $element"
}
}
}
#
# Access examples
#
# Access a block of data
puts "foo\\a = [keylget foo a]"
# Access a nested block of data
puts "foo\\b\\ba = [keylget foo b.ba]"
# Access an individual element, remember that Tcl's list index is 0 based
puts "foo\\c\\cb\\1 = [lindex [keylget foo c.cb] 0]"