perl – 我在这里缺少一些$_的语义?

在剖析期间,我在 List::UtilsBy中遇到了这个功能
sub rev_nsort_by(&@) {
    my $keygen = shift;
    my @keys = map { local $_ = $_[$_]; scalar $keygen->( $_ ) } 0 .. $#_;
    return map { $_[$_] } sort { $keys[$b] <=> $keys[$a] } 0 .. $#_;
}

rev_nsort_by基于一些关键谓词进行反向数字排序,例如:

my @objects = load_objects_from_database();
# sort by rating,highest first
@objects = rev_nsort_by { $_->rating } @objects;

我完全理解为什么rev_nsort_by,如上所示,按照预期工作,但我想知道为什么这么复杂.具体来说,我不知道为什么

my @keys = map { local $_ = $_[$_]; scalar $keygen->( $_ ) } 0 .. $#_;

没有写成

my @keys = map { scalar $keygen->( $_ ) } @_;

看起来功能上与我相当.我错过了$_这里的一些角色行为,其中较长的版本以某种方式支付?

解决方法

这里有一个微妙的边缘案例:内部foreach循环或映射表达式,默认变量$_别名为原始值.例如.
@nums = 1..5;
@foo = map { $_ *= 2 } @nums;
# both @foo and @nums contain 2,4,6,8,10 now.

然而,常量不是有效的左值,所以我们不能这样做

@foo = map { $_ *= 2 } 1,2,3,5;
# Modification of read-only value

@_数组也被别名为原始值,因此可以想象以下边缘情况:

sub buggy (&@) { my $cb = shift; map $cb->($_),@_ };

buggy { $_ *= 2 } 1,3;   # Modification of read-only value attempted
buggy { $_[0] *= 2} 1,3; # ditto

my @array = 1 .. 5;
buggy { $_ *= 2 } @array;    # @array now is 2,10
buggy { $_[0] *= 2 } @array; # ditto

别名是可传递的,所以内部$_ [0]被别名为$_,它被别名为外部$_ [0],它是常量1 / $数组[0]的别名.

那么,本地$_ = $_ [$_]在这里呢?

>它创建一个值的副本,从而避免这种疯狂的混叠行为
>它显示了让$_对回调可见的意图.

确保复制语义(从而避免意外的副作用)对于Perl来说是很自然的,所以这个功能是精心设计的,而不是特别的强化.

(注意:地图{local $_ = $_; …} @_将足以制作副本)

相关文章

忍不住在 PerlChina 邮件列表中盘点了一下 Perl 里的 Web 应用框架(巧的是 PerlBuzz 最近也有一篇相关...
bless有两个参数:对象的引用、类的名称。 类的名称是一个字符串,代表了类的类型信息,这是理解bless的...
gb2312转Utf的方法: use Encode; my $str = "中文"; $str_cnsoftware = encode("utf-8...
  perl 计算硬盘利用率, 以%来查看硬盘资源是否存在IO消耗cpu资源情况; 部份代码参考了iostat源码;...
1 简单变量 Perl 的 Hello World 是怎么写的呢?请看下面的程序: #!/usr/bin/perl print "Hello W...
本文介绍Perl的Perl的简单语法,包括基本输入输出、分支循环控制结构、函数、常用系统调用和文件操作,...