因此,我理解以下代码的作用及其工作原理,但似乎应该有不同的方法来执行此操作:
my @squares = map { $_ > 5 ? ($_ * $_) : () } @numbers;
有没有办法我们可以说:
my @squares = map { if ($_ > 5) then ($_ * $_) } @numbers;
或者我们必须对每个条目都有“规则”,即return return()?
答案 0 :(得分:5)
为此,map
块具有以返回不需要的项目的空列表。虽然您可以使用grep
,
my @squares = map { $_**2 } grep { $_ > 5 } @numbers;
这会失去很多优雅。
如果我们没有指定else
返回值,则if
似乎隐式传递了错误值:
say $_+0 for map{ if($_>5){$_**2} } 3..7;
# 0
# 0
# 0
# 36
# 49
对我们来说没用。
但是我们总是可以编写一个过滤映射来返回我们的块的值,如果它是false,则写入空列表:
sub mapgrep (&@) {
my $cb = shift;
map { local $_ = $_; $cb->($_) || () } @_;
}
my @squares = mapgrep { $_**2 if $_ > 5 } @numbers;
然而,这依赖于副作用,如果不以其他方式处理此状态,则条件返回条件的值。我无法看到明确记录的位置。
(注意:Perl没有关键字then
)。
答案 1 :(得分:5)
map { $_ > 5 ? ($_ * $_) : () }
使用if
代替条件运算符是
map { if ($_ > 5) { $_ * $_ } else { () } }
地图表达式不可能不返回值。它返回最后评估的表达式。如果删除else
子句,那么该表达式就是比较,所以它与你做的相似
map { if ($_ > 5) { $_ * $_ } else { $_ > 5 } }
虽然$_ > 5
只执行一次,所以我猜它更接近
map { ($_ > 5) && ($_ * $_) }
所以是的,你必须对每个条目都有一个规则,因为它是不可能的。