大列表重构

Raku 最近(2015.9)经历的最大的变化就是 Great List Refactor(GLR), 俗称大列表重构。

它还很难解释!但是幸好有某些历史背景能帮助我们。 在 2014 年澳大利亚 Perl Workshop 会议上讨论了很多 GLR 的东西, Patrick Michaud 在它的博客上写了很多关于 GLR 的内容。

GLE 意图强调性能和列表和相关类型操作的一致性问题。改变这样的基本数据类型将会很痛苦。

通常 Perl 5 这样展开列表:

% perl -dE 1 -MData::Dumper

[...snip...]
DB<1> @foos=((1,2),3)

DB<2> say Dumper \@foos
$VAR1 = [
    1,
    2,
    3
];

开始, 很多的 Raku 行为都模仿这种展平行为但是在去年年底的时候, 使用 non-flatterning(非展平)行为以保留原始数据结构的用法越来越多。

这样做和很多边界情况不一致并且 Rakudo 内部大量使用了一种叫做 Parcel 的数据类型, 之后 Parcel 被认为是一个 Bad Idea(糟糕的设计) — 主要是因为性能问题。

2015 年 7 月, Patrick 导入了一个涵盖 “Synopsis 7” 的孪生设计草案, 随后的月份中它变成官方的 S07。

8 月份对于 GLR 是很繁忙的一个月。乔纳森.华盛顿在 Rakudo 仓库下开启了一个单独的 GLR 分支。很多人在那个分支下协同工作。同时在 IRC 频道有俩个机器人, Camelia 和 GLRelia(后者跟踪 GLR 分支)。所以人们能很快尝试并比较在新旧系统中代码的行为。大量的变化和诸如 panda 的软件不得不进行修补以保持功能。

Parcel 数据类型变成了 List(它是不可变的并使用圆括号)而数组是 List 的一个可变子类, 它使用方括号。数组不再拥有隐式的列表展平行为。简单的数组可以使用 .flat方法进行展平。

Pre GLR


my @array = 1,(2,3),4;       # 1 2 3 4
@array.elems.say;            # 4   

Post GLR


my @array = 1,(2,3),4;        # [1 (2 3) 4]
@array.elems.say;             # 3

my @array = (1,(2,3),4).flat; # [1 2 3 4]

把列表滑进(Slip)数组中也是可行的:

my @a = 1, (2, 3).Slip, 4;    # [1 2 3 4]

还有:

my @a = 1, |(2,3), 4;         # [1 2 3 4]

序列(只能够被耗费一次)被引入:

my $grep = (1..4).grep(*>2);  # (3 4)
$grep.^name.say;              # Seq

.cache方法能用于阻止 Seq 的消费。

The Single Argument Rule


传递给诸如 for循环迭代器的参数遵守"the single arg rule", 意思是即使第一眼看上去是以多个参数出现的, 也会被当作单个参数。通常这会让 for 表现得如程序员所期望的那样除了带有尾随逗号的例子。

my @a = 1,2,3;
my ($i, $j);
for (@a) {
    $i++;
}

for (@a,) { # 这实际上是单个元素列表(single element list)
    $j++;
}

say :$i.gist;  # => 3
say :$j.gist;  # => 1

S07 在 2015年9月份被乔纳森-华盛顿重写了。结果就是 S07经历了很多改变。Parcel 被移除了, 重新引入进来, 并且最终又被移除了!

comments powered by Disqus