TCL:如何返回数组?

时间:2012-12-01 23:30:39

标签: arrays return tcl

请在下面找到传递数组的代码片段,操作数组,但我无法返回新版本的数组。

这是片段:

    proc    get_mroute_active { &multicast }   {
        upvar   ${&multicast} MULTICAST ;

                set group   -1 ;
                set src -1 ;
                                set     mcast_group_source_id   -1 ;
                                set     MULTICAST($mcast_group_source_id,id) $mcast_group_source_id ;
                                set     MULTICAST($mcast_group_source_id,mcast_group) $group ;
                                set     MULTICAST($mcast_group_source_id,mcast_source) $src ;

        puts    [array size MULTICAST] ;
    parray  MULTICAST ;
}


array set     multicast { } ;

get_mroute_active [array get multicast] ;
puts    [array size multicast] ;
parray multicast ;

代码的输出是:

3
MULTICAST(-1,id)           = -1
MULTICAST(-1,mcast_group)  = -1
MULTICAST(-1,mcast_source) = -1
0

请您帮忙告诉我如何将“MULTICAST”变量分配给“多播”?

1 个答案:

答案 0 :(得分:5)

简短的回答是:您无法从过程返回数组,因为数组不是值 - 它们是命名值的特殊命名集合。

有几种方法可以解决这个问题:

通常的方法是按名称传递数组,并使被调用的过程修改数组。例如,代码

proc foo {arrayName} {
    upvar 1 $arrayName ary
    incr ary(one)
}
set x(one) 1
foo x
puts $x(one)

将打印“2”,因为过程foo修改了调用者范围内指定数组中的特定值。

注意调用者如何传递数组的名称“x”,而不是“它的值”(因为你不能从数组中提取值;但是见下文),然后被调用的过程将一个局部变量绑定到该数组按名称使用upvar命令。

另一种方法是使用array get and array set命令从数组中提取键和值,并分别使用键和值填充数组。例如:

set x(one) 1
set x(two) 2
array set another [array get x]
parray another

会打印

another(one) = 1
another(two) = 2

给定一个数组的array get命令返回一个平面列表及其键和它们各自的值交错。这样,您可以从过程返回数组的内容,然后让调用者对这些内容执行任何操作,例如,使用array set命令在其中填充另一个数组范围(可能与首先传递给该程序的相同)。

请注意array set具有合并语义:在使用源列表插入/替换其值之前,它不会清空目标数组。

第三种(也可能是最好的)方法是使用dictionaries作为数组的键/值映射,但它们本身就是值,因此它们可以作为其他值自由传递。这需要Tcl 8.5或更高版本(在Tcl 8.4中,您可以以包的形式使用a backport of this code。使用dict s,您可以这样:

proc foo d {
    dict set d one 2
    return $d
}
set x [dict create]
dict set x one 1
set y [foo $x]
puts $y

将打印“one 2”,因为程序修改了原始字典然后返回它,然后调用者将其分配给另一个变量。