Я часто использую этот шаблон в Perl
sub fun {
my $line = $_[0];
my ( $this, $that, $the_other_thing ) = split /\t/, $line;
return { 'this' => $this, 'that' => $that, 'the_other_thing' => $the_other_thing};
}
Очевидно, что я могу упростить этот шаблон, возвращая выходные данные функции, которая преобразует данный массив переменных в карту, где ключи имеют те же имена, что и переменные, например
sub fun {
my $line = $_[0];
my ( $this, $that, $the_other_thing ) = split /\t/, $line;
return &to_hash( $this, $that, $the_other_thing );
}
Это помогает, когда количество элементов увеличивается. Как мне это сделать? Похоже, я мог бы комбинировать PadWalker и замыкания, но я бы хотел сделать это, используя только основной язык.
РЕДАКТИРОВАТЬ: THB предоставил умное решение этой проблемы, но я не проверял его, потому что он обходит много сложных частей (тм). Как бы вы это сделали, если бы хотели полагаться на семантику деструктуризации базового языка и отражать фактические переменные?
EDIT2: вот решение, на которое я намекал, используя PadWalker & closures:
use PadWalker qw( var_name );
# Given two arrays, we build a hash by treating the first set as keys and
# the second as values
sub to_hash {
my $keys = $_[0];
my $vals = $_[1];
my %hash;
@hash{@$keys} = @$vals;
return \%hash;
}
# Given a list of variables, and a callback function, retrieves the
# symbols for the variables in the list. It calls the function with
# the generated syms, followed by the original variables, and returns
# that output.
# Input is: Function, var1, var2, var3, etc....
sub with_syms {
my $fun = shift @_;
my @syms = map substr( var_name(1, \$_), 1 ), @_;
$fun->(\@syms, \@_);
}
sub fun {
my $line = $_[0];
my ( $this, $that, $other) = split /\t/, $line;
return &with_syms(\&to_hash, $this, $that, $other);
}