在像 1 + 2 * 3
这样的表达式中, 2 * 3
被首先计算, 因为中缀操作符 *
的优先级比 +
的优先级高。下面的表中总结了 Perl 6 中 的优先级级别, 从最牢固到最松散:
A Level Examples
N Terms 42 3.14 "eek" qq["foo"] $x :!verbose @$array
L 方法后缀 .meth .+ .? .* .() .[] .{} .<> .«» .:: .= .^ .:
N 自增 ++ --
R 求幂 **
L Symbolic unary ! + - ~ ? | || +^ ~^ ?^ ^
L 乘法 * / % %% +& +< +> ~& ~< ~> ?& div mod gcd lcm
L 加法 + - +| +^ ~| ~^ ?| ?^
L 重复 x xx
X 连结 ~
X Junctive and &
X Junctive or | ^
L Named unary temp let
N Structural infix but does <=> leg cmp .. ..^ ^.. ^..^
C Chaining infix != == < <= > >= eq ne lt le gt ge ~~ === eqv !eqv
X Tight and &&
X Tight or || ^^ // min max
R Conditional ?? !! ff fff
R Item assignment = => += -= **= xx= .=
L Loose unary so not
X Comma operator , :
X List infix Z minmax X X~ X* Xeqv ...
R List prefix print push say die map substr ... [+] [*] any Z=
X Loose and and andthen
X Loose or or xor orelse
X Sequencer <==, ==>, <<==, ==>>
N Terminator ; {...}, unless, extra ), ], }
下面使用的两处 !
符号一般代表任何一对儿拥有相同优先级的操作符, 上表指定的二元操作符的结合性解释如下(其中 A 代表结合性, associativities ):
A Assoc Meaning of $a ! $b ! $c
L left ($a ! $b) ! $c
R right $a ! ($b ! $c)
N non ILLEGAL
C chain ($a ! $b) and ($b ! $c)
X list infix:<!>($a; $b; $)
对于一元操作符, 这解释为:
A Assoc Meaning of !$a!
L left (!$a)!
R right !($a!)
N non ILLEGAL
下面描述的操作符, 默认假定为 left 结合性。
操作符能出现在相对于 term 的几个位置处:
+term prefix (后缀)
term1 + term2 infix (中缀)
term++ postfix (后缀)
(term) circumfix (环缀)
term1[term2] postcircumfix (后环缀)
每个操作符也可以用作子例程。 这样的子例程的名字由操作符的种类, 然后后跟一个冒号,再加上一组引号结构, 引号结构中是组成操作符的符号(s):
infix:<+>(1, 2) # same as 1 + 2
circumfix:«( )»('a', 'b', 'c') # same as ('a', 'b', 'c'), 目前编译错误。
circumfix:<[ ]>('a', 'b', 'c').perl.say # ["a", "b", "c"]
作为一种特殊情况, listop(列表操作符)既能作为 term 又能作为前缀。子例程调用是最常见的列表操作符。其它情况包括元运算中缀操作符 [+]| 1, 2, 3
和 prefix
等 stub 操作符。
定义自定义操作符在 /language/functions#Defining_Operators. 中有涉及。
Term 怎么翻译才合适? 我觉得翻译成 项
更合适, 表明这是一个名词。
分组操作符。
空的分组 ()
创建一个空的 Pracel
。 非空表达式周围的圆括号只是构建了表达式, 而没有额外的语义。
在参数列表中,在参数周围放上圆括号防止了参数被解释为具名参数。
multi sub p(:$a!) { say 'named' }
multi sub p($a) { say 'positional' }
p a => 1; # named
p (a => 1); # positional
Block 或 散列构造器。
如果`{}` 里面的内容看起来像一组 pairs 并且没有 $_
或其它占位符参数,就返回一个散列, 这个散列由逐项逐项的 pair 组成。
否则就返回一个 Block。
注意,这个结构没有重新解析内容; 而里面的内容总是被解析为一组句子(例如,像一个 block), 并且如果后面的分析表明它需要被解析成一个散列, 那么 block 就会被执行并强转为散列。
sub postcircumfix:<[ ]>(@container, **@index,
:$k, :$v, :$kv, :$p, :$exists, :$delete)
:$k
会创建一个 pair, 它是散列中的一个条目。 键是 k
, 键值为 $kv
。 所以, $k
等价于 k
⇒ $k
访问 @container
中的一个或多个元素,即数组索引操作:
my @alphabet = 'a' .. 'z';
say @alphabet[0]; #-> a
say @alphabet[1]; #-> b
say @alphabet[*-1]; #-> z
say @alphabet[100]:exists; #-> False
say @alphabet[15, 4, 17, 11].join; #-> perl
say @alphabet[23 .. *].perl; #-> ("x", "y", "z")
@alphabet[1, 2] = "B", "C";
say @alphabet[0..3].perl #-> ("a", "B", "C", "d")
查看 Subscripts 获取关于该操作符行为的更详细的解释, 还有怎么在自定义类型中实现对它的支持。
sub postcircumfix:<{ }>(%container, **@key,
:$k, :$v, :$kv, :$p, :$exists, :$delete)
访问 %container
的一个或多个元素, 即散列索引操作:
my %color = kiwi => "green", banana => "yellow", cherry => "red";
say %color{"banana"}; #-> yellow
say %color{"cherry", "kiwi"}.perl; #-> ("red", "green")
say %color{"strawberry"}:exists; #-> False
%color{"banana", "lime"} = "yellowish", "green";
%color{"cherry"}:delete;
say %color; #-> banana => yellowish, kiwi => green, lime => green
查看 后环缀 < >
和 后环缀 « »
作为便捷形式, 查看 Subscripts
获取这个操作符行为的更详细解释, 还有怎么在自定义类型中实现对它的支持。
后环缀 { }
的简写形式, 它会引起它的参数。
my %color = kiwi => "green", banana => "yellow", cherry => "red";
say %color<banana>; #-> yellow
say %color<cherry kiwi>.perl; #-> ("red", "green")
say %color<strawberry>:exists; #-> False
这不是一个真正的操作符, 它仅仅是一个在编译时把 < >
变成 {}
后环缀操作符的语法糖。
后环缀 { }
的简写形式。它会引起它的参数,并且 « » 中能进行变量插值。
my %color = kiwi => "green", banana => "yellow", cherry => "red";
my $fruit = "kiwi";
say %color«cherry $fruit».perl; #-> ("red", "green")
这不是一个真正的操作符, 它仅仅是一个在编译时把 « » 变成 {}
后环缀操作符的语法糖。
调用操作符。把调用者当作 Callable
并引用它,它使用圆括号之间的表达式作为参数。
注意,标识符后面直接跟着一对儿圆括号总是被解析为子例程调用。
如果你想要你的对象响应该调用操作符, 你需要实现 CALL-ME
方法。
可变的方法调用。 $invocant.=method
, 脱去语法糖后就是 $invocant = $invocant.method
, 这与 http://doc.raku.org/routine/op%3D 类似。
技术上讲, 这不是一个操作符,而是编译器中特殊情况下的语法。
元方法调用。 ` $invocant.^method` 在 $invocant
的元类身上调用方法。脱去语法糖后, 它就是 $invocant.HOW.method($invocant, …)
。 查看 HOW 获取更多信息。
技术上讲, 这不是一个操作符,而是编译器中特殊情况下的语法。
有可能被调用`的方法调用。如果有名为 `method
的方法, $invocant.?method
就在 $invocant
上调用 method
方法。否则它就返回 Nil
。
技术上讲, 这不是一个操作符,而是编译器中特殊情况下的语法。
$invocant.+method ` 从 `$invocant
身上调用所有叫做 method
的方法, 并返回一个 Parcel 作为结果。 如果没有找到这个名字的方法, 就会死掉。
技术上讲, 这不是一个操作符,而是编译器中特殊情况下的语法。
$invocant.*method
` 从 $invocant
身上调用所有叫做 method
的方法, 并返回一个 Parcel 作为结果。 如果没有找到这个名字的方法,则返回一个空的 Parcel。
技术上讲, 这不是一个操作符,而是编译器中特殊情况下的语法。
大多数情况下, 可以在后缀或后环缀前面放上一个点:
@a[1, 2, 3];
@a.[1, 2, 3]; # Same
这对于视觉清晰或简洁很有帮助。例如,如果对象的属性是一个函数,在属性名后面放置一对儿圆括号会变成方法调用的一部分。 所以要么使用两对儿圆括号, 要么在圆括号前面放上一个点来阻止方法调用。
class Operation {
has $.symbol;
has &.function;
}
my $addition = Operation.new(:symbol<+>, :function{ $^a + $^b });
say $addition.function()(1, 2); # 3
或者
say $addition.function.(1,2); # 3
然而,如果后缀是一个标识符, 那么它会被解释为一个普通的方法调用。
1.i # No such method 'i' for invocant of type 'Int'
技术上讲, 这不是一个操作符,而是编译器中特殊情况下的语法。
前缀能够像方法那样, 使用冒号对儿标记法来调用。例如:
my $a = 1;
say ++$a; # 2
say $a.:<++>; # 3
技术上讲, 这不是一个操作符,而是编译器中特殊情况下的语法。
multi sub prefix:<++>($x is rw) is assoc<none>
把它的参数增加 1, 并返回增加后的值。
my $x = 3;
say ++$x; # 4
say $x; # 4
它的工作原理是在它的参数身上调用 succ
方法, 这可以让自定义类型自由地实现它们自己的增量语义。
multi sub prefix:<-->($x is rw) is assoc<none>
把它的参数减少 1, 并返回减少后的值。
my $x = 3;
say --$x; # 2
say $x; # 2
它的工作原理是在它的参数身上调用 pred
方法, 这可以让自定义类型自由地实现它们自己的减量语义。
multi sub postfix:<++>($x is rw) is assoc<none>
把它的参数增加 1, 并返回`unincremented`的那个值。
my $x = 3;
say $x++; # 3
say $x; # 4
它的工作原理是在它的参数身上调用 succ
方法, 这可以让自定义类型自由地实现它们自己的增量语义。
注意这并不一定返回它的参数。 例如,对于未定义的值, 它返回 0:
my $x;
say $x++; # 0
say $x; # 1
multi sub prefix:<?>(Mu) returns Bool:D
布尔上下文操作符。
通过在参数身上调用 Bool
方法强制它的参数为 Bool。注意, 这会使 Junctions 失效。
multi sub prefix:<!>(Mu) returns Bool:D
否定的布尔上下文操作符。
通过在参数身上调用 Bool
方法强制它的参数为 Bool, 并返回结果的否定值。注意, 这会使 Junctions 失效。
multi sub prefix:<+>(Any) returns Numeric:D
Numeric 上下文操作符。
通过在参数身上调用 Numeric 方法强制将参数转为 Numeric 类型。
multi sub prefix:<->(Any) returns Numeric:D
否定的 Numeric 上下文操作符。
通过在参数身上调用 Numeric 方法强制将参数转为 Numeric 类型, 并返回结果的否定值。
将 Capture, Enum, Pair, List, Parcel, EnumMap 和 Hash 展平到参数列表中。
(在 Rakudo 中,这不是作为一个合适的操作符来实现的,而是编译器中的一种特殊情况, 这意味着它只对参数列表有效,而非在任意代码中都有效。)
multi sub prefix:<+^>(Any) returns Int:D
Integer bitwise negation
整数按位取反。
将参数强转为 Int 类型并对结果按位取反, 假设两者互补。
multi sub infix:<*>(Any, Any) returns Numeric:D
把两边的参数都强转为 Numeric 并把它们相乘。 结果是一个更宽的类型。 查看 Numeric 获取更详细信息。
multi sub infix:</>(Any, Any) returns Numeric:D
把两边的参数都强制为 Numeric, 并用左边除以右边的数。整数相除返回 Rat, 否则返回"更宽类型” 的结果。
multi sub infix:<%>($x, $y) return Numeric:D
模操作符。首先强制为 Numeric。
通常,下面的等式是成立的:
$x % $y == $x - floor($x / $y) * $y
multi sub infix:<+&>($a, $b) returns Int:D
Numeric 按位 AND
。 把两个参数都强转为 Int 并执行按位 AND 操作,假定两者是互补的。
multi sub infix:<gcd>($a, $b) returns Int:D
强制两个参数都为 Int 并返回最大公分母(greatest common denominator)。
proto sub infix:<x>(Any, Any) returns Str:D
multi sub infix:<x>(Any, Any)
multi sub infix:<x>(Str:D, Int:D)
把 $a
强转为 Str , 把 $b
强转为 Int, 并重复字符串 $b
次。 如果 $b ⇐ 0
则返回空字符串。
say 'ab' x 3; # ababab
say 42 x 3; # 424242
sub infix:<does>(Mu $obj, Mu $role) is assoc<none>
在运行时把 $role
混合进 $obj
中。 要求 $obj
是可变的。
参数 $role
不一定要求是一个 role, 它可以表现的像是一个 role, 例如枚举值。
sub infix:<but>(Mu $obj, Mu $role) is assoc<none>
把 $role
混合进 $obj
并创建一个 $obj
的副本。因为 $obj
是不能修改的,但是能使用 mixins 用于创建不可变值。
参数 $role
不一定要求是一个 role, 它可以表现的像是一个 role, 例如枚举值。
proto sub infix:<cmp>(Any, Any) returns Order:D is assoc<none>
multi sub infix:<cmp>(Any, Any)
multi sub infix:<cmp>(Real:D, Real:D)
multi sub infix:<cmp>(Str:D, Str:D)
multi sub infix:<cmp>(Enum:D, Enum:D)
multi sub infix:<cmp>(Version:D, Version:D)
一般的, “智能的” 三路比较器。
比较字符串时使用字符串语义, 比较数字时使用数字语义, 比较 Pair 对象时, 先比较键, 再比较值,等等。
if $a eqv $b, then $a cmp $b always returns Order::Same.
say (a => 3) cmp (a => 4); # Less
say 4 cmp 4.0; # Same
say 'b' cmp 'a'; # More
proto sub infix:<leg>($a, $b) returns Order:D is assoc<none>
multi sub infix:<leg>(Any, Any)
multi sub infix:<leg>(Str:D, Str:D)
字符串三路比较器。 leg 是 less, equal 还有 greater 的简写形式?
把两个参数都强转为 Str
, 然后按照字母次序比较。
say 'a' leg 'b'; Less
say 'a' leg 'a'; Same
say 'b' leg 'a'; More
multi sub infix:«<=>»($a, $b) returns Order:D is assoc<none>
Numeric 三路比较器。
把两个参数强转为 Real, 并执行数值比较。
proto sub infix:<==>($, $) returns Bool:D is assoc:<chain>
multi sub infix:<==>(Any, Any)
multi sub infix:<==>(Int:D, Int:D)
multi sub infix:<==>(Num:D, Num:D)
multi sub infix:<==>(Rational:D, Rational:D)
multi sub infix:<==>(Real:D, Real:D)
multi sub infix:<==>(Complex:D, Complex:D)
multi sub infix:<==>(Numeric:D, Numeric:D)
强转两个参数为 Numeric(如果必要), 并�返回 True 如果它们相等。
proto sub infix:<!=>(Mu, Mu) returns Bool:D is assoc<chain>
强转两个参数为 Numeric(如果必要), 并�返回 True 如果它们不相等。
proto sub infix:«<»(Any, Any) returns Bool:D is assoc<chain>
multi sub infix:«<»(Int:D, Int:D)
multi sub infix:«<»(Num:D, Num:D)
multi sub infix:«<»(Real:D, Real:D)
强转两个参数为 Real (如果必要), 并返回 True 如果第一个参数小于第二个参数。
proto sub infix:«<=»(Any, Any) returns Bool:D is assoc<chain>
multi sub infix:«<=»(Int:D, Int:D)
multi sub infix:«<=»(Num:D, Num:D)
multi sub infix:«<=»(Real:D, Real:D)
强转两个参数为 Real (如果必要), 并返回 True 如果第一个参数小于第二个参数。
proto sub infix:«>»(Any, Any) returns Bool:D is assoc<chain>
multi sub infix:«>»(Int:D, Int:D)
multi sub infix:«>»(Num:D, Num:D)
multi sub infix:«>»(Real:D, Real:D)
强转两个参数为 Real (如果必要), 并返回 True 如果第一个参数大于第二个参数。
proto sub infix:«>=»(Any, Any) returns Bool:D is assoc<chain>
multi sub infix:«>=»(Int:D, Int:D)
multi sub infix:«>=»(Num:D, Num:D)
multi sub infix:«>=»(Real:D, Real:D)
强转两个参数为 Real (如果必要), 并返回 True 如果第一个参数大于或等于第二个参数。
proto sub infix:<eq>(Any, Any) returns Bool:D is assoc<chain>
multi sub infix:<eq>(Any, Any)
multi sub infix:<eq>(Str:D, Str:D)
强转两个参数为 Str(如果必要), 并返回 True 如果第一个参数等于第二个参数。
助记法: equal
proto sub infix:<ne>(Mu, Mu) returns Bool:D is assoc<chain>
multi sub infix:<ne>(Mu, Mu)
multi sub infix:<ne>(Str:D, Str:D)
强转两个参数为 Str(如果必要), 并返回 False 如果第一个参数等于第二个参数。
助记法: not equal
proto sub infix:<gt>(Mu, Mu) returns Bool:D is assoc<chain>
multi sub infix:<gt>(Mu, Mu)
multi sub infix:<gt>(Str:D, Str:D)
强转两个参数为 Str(如果必要), 并返回 True 如果第一个参数大于第二个参数。
助记法: greater than
proto sub infix:<ge>(Mu, Mu) returns Bool:D is assoc<chain>
multi sub infix:<ge>(Mu, Mu)
multi sub infix:<ge>(Str:D, Str:D)
强转两个参数为 Str(如果必要), 并返回 True 如果第一个参数大于第二个参数。
助记法: greater or equal
proto sub infix:<lt>(Mu, Mu) returns Bool:D is assoc<chain>
multi sub infix:<lt>(Mu, Mu)
multi sub infix:<lt>(Str:D, Str:D)
强转两个参数为 Str(如果必要), 并返回 True 如果第一个参数小于第二个参数。
助记法: less than
proto sub infix:<le>(Mu, Mu) returns Bool:D is assoc<chain>
multi sub infix:<le>(Mu, Mu)
multi sub infix:<le>(Str:D, Str:D)
强转两个参数为 Str(如果必要), 并返回 True 如果第一个参数小于或等于第二个参数。
助记法: less or equal
proto sub infix:<before>(Any, Any) returns Bool:D is assoc<chain>
multi sub infix:<before>(Any, Any)
multi sub infix:<before>(Real:D, Real:D)
multi sub infix:<before>(Str:D, Str:D)
multi sub infix:<before>(Enum:D, Enum:D)
multi sub infix:<before>(Version:D, Version:D)
一般的排序, 使用�和 cmp 相同的语义。 如果第一个参数小于第二个参数则返回 True。
proto sub infix:<after>(Any, Any) returns Bool:D is assoc<chain>
multi sub infix:<after>(Any, Any)
multi sub infix:<after>(Real:D, Real:D)
multi sub infix:<after>(Str:D, Str:D)
multi sub infix:<after>(Enum:D, Enum:D)
multi sub infix:<after>(Version:D, Version:D)
一般的排序, 使用�和 cmp 相同的语义。 如果第一个参数大于第二个参数则返回 True。
proto sub infix:<eqv>(Any, Any) returns Bool:D is assoc<chain>
proto sub infix:<eqv>(Any, Any)
等值操作符。如果两个参数在结构上相同就返回 True。例如, 相同类型(并且递归)包含相同的值。
say [1, 2, 3] eqv [1, 2, 3]; # True
say Any eqv Any; # True
say 1 eqv 2; # False
say 1 eqv 1.0; # False
对于任意对象使用默认的 eqv 操作是不可能的。例如, eqv 不认为同一对象的两个实例在结构上是相等的:
class A {
has $.a;
}
say A.new(a => 5) eqv A.new(a => 5); #=> False
要得到这个类的对象相等(eqv)语义, 需要实现一个合适的中缀 eqv 操作符:
class A {
has $.a;
}
multi infix:<eqv>(A $l, A $r) { $l.a eqv $r.a }
say A.new(a => 5) eqv A.new(a => 5); #=> True
proto sub infix:<===>(Any, Any) returns Bool:D is assoc<chain>
proto sub infix:<===>(Any, Any)
值相等。如果两个参数都是同一个对象则返回 True。
class A { };
my $a = A.new;
say $a === $a; # True
say A.new === A.new; # False
say A === A; # True
对于值的类型, ===
表现的和 eqv 一样:
say 'a' === 'a'; # True
say 'a' === 'b'; # False
# different types
say 1 === 1.0; # False
===
使用 WHICH
方法来获取对象相等, 所以所有的值类型必须重写方法 WHICH
。
proto sub infix:<=:=>(Mu \a, Mu \b) returns Bool:D is assoc<chain>
multi sub infix:<=:=>(Mu \a, Mu \b)
容器相等。返回 True 如果两个参数都绑定到同一个容器上。 如果它返回 True, 那通常意味着修改一个参数也会同时修改另外一个。
my ($a, $b) = (1, 3);
say $a =:= $b; # False
$b = 2;
say $a; # 1
$b := $a;
say $a =:= $b; # True
$a = 5;
say $b; # 5
在布尔上下文中返回第一个求值为 True 的参数, 否则返回最后一个参数。
注意这是短路操作符,如果其中的一个参数计算为 true 值, 那么该参数右侧的值绝不会被计算。
sub a { 0 }
sub b { 1 }
sub c { die "never called" };
say a() || b() || c(); # 1
返回第一个值为 true 的参数如果只有一个的话, 否则返回 Nil。只要找到两个值为 true 的参数就发生短路。
say 0 ^^ 42; # 42
say 0 ^^ 42 ^^ 1 ^^ die 8; # (empty line)
注意, 这个操作符的语义可能不是你假想的那样: infix ^^ 翻到它找到的第一个 true 值, 找到第二个 true 值后永远地反转为 Nil 值, 不管还有多少 true 值。(换句话说,它的语义是”找到一个真值”, 而不是布尔起奇偶校验语义)
三目操作符, 条件操作符。
$condition ?? $true !! $false
计算并返回 $true
表达式,如果 $condition
为真的话。 否则计算并返回 $false
分支。
sub infix:<ff>(Mu $a, Mu $b)
Flipflop operator. 触发器操作符。
把两个参数都跟 $
进行比较(即,$
~~ $a
和 $_ ~~ $b
)。求值为 False 直到左侧的智能匹配为真, 这时,它求值为真, 直到右侧的智能匹配为真。
实际上,左边的参数是"开始”条件, 右侧的参数是”停止” 条件。 这种结构一般用于收集只在特定区域的行。 例如:
my $excerpt = q:to/END/;
Here's some unimportant text.
=begin code
This code block is what we're after.
We'll use 'ff' to get it.
=end code
More unimportant text.
END
my @codelines = gather for $excerpt.lines {
take $_ if "=begin code" ff "=end code"
}
# this will print four lines,
# starting with "=begin code" and ending with "=end code"
say @codelines.join("\n");
匹配开始条件之后,操作符会继续将停止条件与 $_
进行匹配, 如果成功就做相应地表现。在这个例子中, 只有第一个元素被打印了:
for <AB C D B E F> {
say $_ if /A/ ff /B/; # prints only "AB"
}
如果你想测试开始条件, 并且没有结束条件, *
能用作 “停止” 条件。
for <A B C D E> {
say $_ if /C/ ff *; # prints C, D, and E
}
对于 sed-like 版本, 在开始条件匹配成功之后,它不会使用停止条件与 $_
进行匹配。
这个操作符不能被重载, 因为它被编译器特殊处理过。
sub infix:<^ff>(Mu $a, Mu $b)
像 ff 那样工作,除了它不会在条目匹配开始条件时返回真。(包括匹配停止条件的条目)
一个比较:
my @list = <A B C>;
say $_ if /A/ ff /C/ for @list; # prints A, B, and C
say $_ if /A/ ^ff /C/ for @list; # prints B and C
sed-like 版本 可以在 ^fff 中找到.
这个操作符不能被重载, 因为它被编译器特殊处理过。
sub infix:<ff^>(Mu $a, Mu $b)
像 ff 那样工作,除了它不会在条目匹配停止条件时返回真。(包括第一次匹配开始条件的条目)
my @list = <A B C>;
say $_ if /A/ ff /C/ for @list; # prints A, B, and C
say $_ if /A/ ff^ /C/ for @list; # prints A and B
sed-like 版本 可以在 fff^ 中找到.
这个操作符不能被重载, 因为它被编译器特殊处理过。
sub infix:<^ff^>(Mu $a, Mu $b)
像 ff 那样工作,除了它不会在条目匹配停止条件时返回真, 也不会在条目匹配开始时返回真。(或者两者)
my @list = <A B C>;
say $_ if /A/ ff /C/ for @list; # prints A, B, and C
say $_ if /A/ ^ff^ /C/ for @list; # prints B
sed-like 版本 可以在 fff 中找到.
这个操作符不能被重载, 因为它被编译器特殊处理过。
sub infix:<fff>(Mu $a, Mu $b)
执行 sed-like 那样的 flipflop 操作,在其中,它返回 False 直到左侧的参数与 $
智能匹配, 并且在那之后返回 True 直到右侧的参数和 $
智能匹配。
像 ff 那样工作, 除了它每次调用只尝试一个参数之外。即, 如果 $
和左侧的参数智能匹配, fff 随后不会尝试将同一个 $
和右侧的参数进行匹配。
for <AB C D B E F> {
say $_ if /A/ fff /B/; # Prints "AB", "C", "D", and "B"
}
对于 non-sed-like 版本, 查看 ff
.
这个操作符不能被重载, 因为它被编译器特殊处理过。
sub infix:<^fff>(Mu $a, Mu $b)
像 fff那样, 除了它对于左侧的匹配不返回真之外。
my @list = <A B C>;
say $_ if /A/ fff /C/ for @list; # prints A, B, and C
say $_ if /A/ ^fff /C/ for @list; # prints B and C
对于 non-sed 版本, 查看 ^ff
.
这个操作符不能被重载, 因为它被编译器特殊处理过。
sub infix:<fff^>(Mu $a, Mu $b)
像 fff 那样, 除了它对于右侧的匹配不返回真之外。
my @list = <A B C>;
say $_ if /A/ fff /C/ for @list; # prints A, B, and C
say $_ if /A/ fff^ /C/ for @list; # prints A and B
对于 non-sed 版本, 查看 ff^
.
这个操作符不能被重载, 因为它被编译器特殊处理过。
sub infix:<=>(Mu $a is rw, Mu $b)
Item 赋值.
把 = 号右侧的值放入左侧的容器中。 它真正的语义是由左侧的容器类型决定的。
(注意 item 赋值和列表赋值的优先级级别不同, 并且等号左侧的语法决定了等号是被解析为 item 赋值还是列表赋值操作符)。
sub infix:<Z>(**@lists) returns List:D is assoc<chain>
Zip operator.
Z 像一个拉链那样把列表插入进来, 只要第一个输入列表耗尽就停止:
say (1, 2 Z <a b c> Z <+ ->).perl; # ((1, "a", "+"), (2, "b", "-")).list
Z
操作符也作为元操作符存在, 此时内部的 parcels 被应用了元操作符的列表替换:
say 100, 200 Z+ 42, 23; # 142, 223
say 1..3 Z~ <a b c> Z~ 'x' xx 3; # 1ax 2bx 3cx
sub infix:<X>(**@lists) returns List:D is assoc<chain>
从所有列表创建一个外积。最右边的元素变化得最迅速。
1..3 X <a b c> X 9
# produces (1, 'a', 9), (1, 'b', 9), (1, 'c', 9),
(2, 'a', 9), (2, 'b', 9), (2, 'c', 9),
(3, 'a', 9), (3, 'b', 9), (3, 'c', 9)
X 操作符也可以作为元操作符, 此时内部的 parcels 被应用了元操作符的列表的值替换:
1..3 X~ <a b c> X~ 9
# produces '1a9', '1b9', '1c9',
'2a9', '2b9', '2c9',
'3a9', '3b9', '3c9'
multi sub infix:<...>(**@) is assoc<list>
multi sub infix:<...^>(**@) is assoc<list>
序列操作符是一个用于产生惰性列表的普通操作符。
它可以有一个初始元素和一个生成器在 …
的�左侧, 在右侧是一个端点。
序列操作符会使用尽可能多的参数来调用生成器。参数会从初始元素和已生成元素中获取。
默认的生成器是 .succ
或 .pred
, 取决于末端怎么比较:
say 1 ... 4; # 1 2 3 4
say 4 ... 1; # 4 3 2 1
say 'a' ... 'e'; # a b c d e
say 'e' ... 'a'; # e d c b a
(Whatever) 末端生成一个无限序列,使用的是默认的生成器
.succ
。
say (1 ... *)[^5]; # 1 2 3 4 5
自定义生成器是在 …
操作符之前的最后一个参数。下面这个自定义生成器接收两个参数, 生成了斐波纳契数。
say (1, 1, -> $a, $b { $a + $b } ... *)[^8]; # 1 1 2 3 5 8 13 21
# same but shorter
say (1, 1, *+* ... *)[^8]; # 1 1 2 3 5 8 13 21
当然自定义生成器也能只接收一个参数。
say 5, { $_ * 2 } ... 40; # 5 10 20 40
生成器的参数个数至少要和初始元素的个数一样多。
如果没有生成器,并且有不止一个初始元素,所有的初始元素都是数值,那么序列操作符会尝试推导出生成器。它知道数学和几何序列。
say 2, 4, 6 ... 12; # 2 4 6 8 10 12
say 1, 2, 4 ... 32; # 1 2 4 8 16 32
如果末端不是 *
, 它会和每个生成的元素进行智能匹配,当智能匹配成功的时候序列就被终止。对于 …
操作符, 会包含最后一个元素, 对于 …^
操作符,会排除最后的那个元素。
这允许你这样写:
say 1, 1, *+* ...^ *>= 100;
来生成所有直到 100 但不包括 100 的斐波纳契数。
…
操作符还会把初始值看作”已生成的元素”,所以它们也会对末端进行检查:
my $end = 4;
say 1, 2, 4, 8, 16 ... $end;
# outputs 1 2 4
列表赋值。 它真正的语义是由左侧的容器类型决定的。查看 Array 和 Hash 获取普通案例。
item 赋值和列表赋值的优先级级别不同, 并且等号左侧的语法决定了等号是被解析为 item 赋值还是列表赋值操作符。
绑定。 而 $x = $y
是把 $y
中的值放到 $x
里面, $x := $y
会让 $x
和 $y
引用同一个值。
my $a = 42;
my $b = $a;
$b++;
say $a;
这会输出 42, 因为 $a 和 $b 都包含了数字 42, 但是容器是不同的。
my $a = 42;
my $b := $a;
$b++;
say $a;
这会打印 43, 因为 $b
和 $a
都代表着`同一个对象`。
这是yada, yada, yada 操作符 或 stub 操作符。如果它在子例程或类型中是唯一的语句,它会把子例程或类型标记为 stub(这在预声明类型和组成 roles 上下文中是有意义的)
如果 …
语句被执行了, 它会调用 &fail
, 伴随着默认的消息 stub 代码的执行。
如果它在子例程或类型中是唯一的语句,它会把子例程或类型标记为 stub(这在预声明类型和组成 roles 上下文中是有意义的)
如果 !!!
语句被执行了, 它会调用 &die
, 伴随着默认的消息 stub 代码的执行。