I want to pass variable arguments obtained in one function to other function but I am not able to do so. Function gets even number of variable arguments and then it has to be converted in array. Below is the example.
Procedure abc1 gets two arguments (k k) and not form abc1 procedure these have to be passed to proc abc where list to array conversion it to be done. List to array conversion works in proc1 i.e. abc1 but not in second proc i.e. abc
Error obtained is given below
proc abc {args} {
puts "$args"
array set arg $args
}
proc abc1 {args} {
puts "$args"
array set arg $args
set l2 [array get arg]
abc $l2
}
abc1 k k
abc k k
Output:
k k
{k k}
list must have an even number of elements
while executing
"array set arg $l1"
(procedure "abc" line 8)
invoked from within
"abc $l2"
(procedure "abc1" line 5)
invoked from within
"abc1 k k"
(file "vfunction.tcl" line 18)
Best Solution: Expansion Substitution
The right approach is to ensure that the outer procedure (in stack terms) calls the inner one correctly; if multiple arguments are expected, that’s what should be supplied. With the advent of Tcl 8.5, that’s trivially done with a little language syntax called an expansion substitution:
All the
{*}does is say that the rest of the word should be broken up (using list syntax rules) and used as multiple arguments instead of Tcl’s default “one visual word forms a single word” rules. It’s ideal for this.Old Solution: Eval Command
If you’re still using old versions of Tcl for some reason (i.e., Tcl 8.4 or before) then you use the
evalcommand instead of the above syntax:There are some somewhat-more-efficient approaches to the above
eval, which you might see in older code; for example:But really they’re all rendered obsolete by
abc {*}$l2which is shorter, simpler to write, and faster. (It’s just not available in 8.4 or before, and too many deployments have yet to upgrade.) Use the expansion syntax if you can. Indeed, idiomatic Tcl code for 8.5 and later hardly ever needseval; the extent to which this has proved true has even been rather surprising to the language’s maintainers.