Perl Tkx:如何将变量作为参数传递给按钮的回调

时间:2015-09-29 14:46:13

标签: perl tkx

鉴于此Perl / Tkx代码片段:

var selected_val = $.map($("select[class='Candidates_dd']:selected"), function(a)
    {
        return a.value;
    }).join(',');

alert(selected_val);

(在真实程序中,@itemList是从用户可编辑的配置文件中填充的。)

我确实看到了两个标有' name1'和' name2'。但是,当我点击任一按钮时,似乎传递给回调的参数始终为@itemList = ({'attrib1' => 'name1', 'attrib2' => 'value1'}, {'attrib1' => 'name2', 'attrib2' => 'value2'}); $row = 0; foreach $item (@itemList) { push(@btn_list, new_ttk__button(-text => $item->{'attrib1'}, -command => sub {do_something($item->{'attrib2'});})); $btn_list[-1]->g_grid(-column => 0, -row => $row); $row++; } ;即' attrib2' @itemList数组的最后一个元素。我想要的是第一次按键调用$itemList[1]->{'attrib2'}和第二次调用do_something($itemList[0]->{'attrib2'}

我做错了什么,拜托并谢谢你?

1 个答案:

答案 0 :(得分:3)

你在Perl中遇到了for循环的一个微妙特征。首先解决方案:在for循环中使用my。然后$item将能够在您稍后在循环中声明的匿名子中创建一个正确的闭包。

for my $item (@itemlist) {
    push(@btn_list, new_ttk__button(
                       -text => $item->{'attrib1'}, 
                       -command => sub {do_something($item->{'attrib2'});}));
    $btn_list[-1]->g_grid(-column => 0, -row => $row);
    $row++;
}

进一步说明:Perl隐式本地化for循环的主题变量。如果在for循环中不使用my,则循环将使用包变量的本地化版本。这使您的代码等同于:

package main;
$main::item = undef;
@itemList = ({'attrib1' => 'name1', 'attrib2' => 'value1'},
             {'attrib1' => 'name2', 'attrib2' => 'value2'});
$row = 0;
foreach (@itemList) {
    local $main::item = $_;
    push(@btn_list, new_ttk__button(
        -text => $main::item->{'attrib1'},
        -command => sub {do_something($main::item->{'attrib2'});}));
    $btn_list[-1]->g_grid(-column => 0, -row => $row);
    $row++;
}
# at the end of the loop, value of $main::item restored to  undef

你的匿名sub仍然引用$main::item包变量,无论变量在调用这些子例程时保持的值,可能是undef

更短的解决方案:use strict

额外的概念验证。试着猜出以下程序的输出:

@foo = ( { foo => 'abc', bar => 123 },
         { foo => 'def', bar => 456 } );

my @fn;
foreach $foo (@foo) {
    push @fn, sub { "42" . $foo->{bar} . "\n" };
}
foreach my $foo (@foo) {
    push @fn, sub { "19" . $foo->{foo} . "\n" };
}
print $_->() for @fn;

以下是答案:

42
42
19abc
19def