在 Perl 6 的方法和函数调用中使用冒号

Use of colon in method and function calls in Perl 6

我想知道冒号与 Perl 6 中的方法和函数调用有什么关系。 作为记录,我使用的是基于 MoarVM 版本 2015.05 构建的 perl6 版本 2015.05-55-gd84bbbc。

我刚刚在Perl6 spec test (S32-io)中看到以下内容(我添加了评论):

$fh.print: "0123456789A";   # prints '0123456789A' to the file

据我所知,这相当于:

$fh.print("0123456789A");   # prints '0123456789A' to the file

这两个似乎都采用多个参数并且可以很好地展平列表:

$fh.print: "012", "345", "6789A";   # prints '0123456789A' to the file
$fh.print("012", "345", "6789A");   # prints '0123456789A' to the file

my @a = <012 345 6789A>;

$fh.print(@a);   # prints '0123456789A' to the file
$fh.print: @a;   # prints '0123456789A' to the file

这两种不同的语法一定是有原因的。是否有任何理由使用一种或另一种语法?

我还注意到,当用作方法时,我们必须使用 :() 进行打印:

$fh.print(@a);   # Works
$fh.print: @a;   # Works!
$fh.print @a;    # ERROR!

在函数 print 中使用冒号时还有一些有趣的行为。在这种情况下,:() 不等价:

print @a;  # Prints '0123456789A' (no newline, just like Perl 5)
print(@a); # Ditto
print: @a; # Prints '012 345 6789A' followed by a newline (at least in REPL)

print  @a, @a; # Error (Two terms in a row)
print: @a, @a; # Prints '012 345 6789A 012 345 6789A' followed by a newline (in REPL) 

然后我尝试在脚本文件中使用打印。这适用于打印到标准输出:

print @a;

但是,这不会打印到标准输出:

print: @a, @a;

但是方法版本工作正常:

$fh.print: @a, @a; # Prints '0123456789A0123456789A' to the file

我觉得我差不多明白了,但是我无法表达出来。有人可以解释一下这些使用印刷品的方式吗?此外,这些行为是否会因 Great List Refactor 而改变?

使用冒号而不是括号的一个主要原因是它可以通过删除一组括号来帮助整理代码。否则它们完全相同。

当你有 print: @a 时,你真正在做的是在线上贴上标签,让 @a 掉线。在 REPL 中将使用值调用 say

如果您在方法调用中不使用圆括号或冒号,则该方法将在没有参数的情况下被调用。


如果使用冒号,则可以调换方法和调用者的顺序。

say $*ERR: 'hello world'; # $*ERR.say('hello world')

进一步澄清 - 使用冒号代替括号是一种替代方法调用语法,而不是一种替代函数调用语法。将函数或块作为参数传递给方法时非常方便。使用 grep 或 map 是常见的例子 - 考虑;

@measurements.map( { check_accuracy($_); fail "bad value" if $_ < 0 }  );

请注意,我在结束卷曲之后放置了一个额外的 space 以尝试将我传递的函数与关闭方法调用所需的必要语法分开。这段代码的 reader 完全理解整个块是调用的一个参数,通常关注块的内容将要实现什么 - 然后遇到我所说的 "Lone Paren" -就像独行侠总是和他信赖的坐骑 Silver 一起出现一样,Lone Paren 几乎总是和他信赖的坐骑 Semicolon 在一起。

为什么不把它们都扔掉?

如果语句以结束卷曲结束,则替代方法调用语法与可选的分号相结合,使之成为可能;

@measurements.map:  { check_accuracy($_);  fail "bad value" if $_ < 0 }

好多了。
虽然替代方法调用格式是另一种需要学习的语法,但它很容易理解,并且在某些情况下(如上),可以使生成的代码更清晰、更易于阅读。