使用raku的subsets和multiple辨别年龄

举个例子, 假设 person 有一个 age 属性. 我能写一个 multimethod, 让它接收一个 person 作为参数, 并返回这样的结果吗:

return "child"  if age < 16;
return "adult"  if 16 <= age < 66;
return "senior" if age >= 66;
class Person {
    has Int $.age;
    has Str $.name;
}

这仅仅定义了一个拥有两个属性, 叫做 Person 的类. age 必须是 Int 型, name 必须是 Str 型. . 语法会生成一个只读访问器, 以使我们能从类的外部访问 getter 方法.

现在我们来定义一个 age-group multi 来告诉一个 person 属于哪个 age-group:

multi age-group ($person where (*.age < 16)  ) { "child" }
multi age-group ($person where (*.age >= 66) ) { "senior"}
multi age-group ($person)                      { "adult" }

where从句给参数添加了一个约束, 这个约束告诉参数必须匹配这个参数右边的东西.这用于区别将要选取的 multi. where从句可以是一个 regex, 类型, 一个确切的值, 一个断言 block,或者一些其它东西.

*.age < 16 部分可能看起来更让人迷惑. 星号是什么? 星号是一个特殊的值, 叫做 Whatever. 它通常在给定情况下满足你的需求. 在智能匹配中, 它总是匹配, 所以你可以在 given/when block 中将它用作默认值. 但是 Whatever 最有用的地方之一是创建匿名 block. 对于大部分操作符, 如果你在 Whatever 上执行它们, 它会产生一个匿名 block 并使用它们的参数执行操作符. 如果一个表达式中有多个 Whatever, 则生成的匿名 block 会有多个参数对应于相应的 Whatever 位置.

例如, + 1 产生一个 block,使参数的值加1. + * 产生一个 block 使它的两个参数相加. 这个例子中, 我们调用 Whatever 的 age方法, 并询问它是否小于 16. 我们能用其它几种方式达到同样的效果, 但是更啰嗦:

sub ($person) { $person.age < 16 }
-> $person    { $person.age < 16 }
{ .age < 16 }

但是对于像这种简单的操作, Whatever 通常比其它方式更易读也更简洁. 不幸的是, 在参数列表的 where 从句中, 你需要使用括号括起很多复杂的表达式, 包括 Whatever block.

现在让我们在 Rakudo 的 REPL 中试试它吧:

> age-group Person.new(:name<timmy>, :age(10))
child
> age-group Person.new(:name<john>, :age(23))
adult
> age-group Person.new(:name<ezekiel>, :age(89))
senior

目前为止, 很好. 但是如果我们意外地传递了一个 age 而不是 Person 给 age-group 呢?

> age-group 15
Method 'age' not found for invocant of class 'Int'

我们能指定只有 Person 对于 age-group 是合法的:

multi age-group (Person $person where (*.age < 16)) { "child" }
multi age-group (Person $person where (*.age >= 66)) { "senior" }
multi age-group (Person $person) { "adult" }

这正确地处理了 Person 问题. 调用带有 age 参数的 age-group 会怎样呢?

> age-group 15
No applicable candidates found to dispatch to for 'age-group'. Available candidates are:
:(Person $person where ({ ... }))
:(Person $person where ({ ... }))
:(Person $person)

看起来更好. 假如我们允许询问 age 所属的 age-group 呢?

我们能重写 age-group 的 Person 变体, 接收 Int 类型的 age, 并写一个单个的 Person 变体来调用 age-group:

multi age-group(Int $age where (* < 16)  ) { "child"  }
multi age-group(Int $age where (* >= 66) ) { "senior" }
multi age-group(Int $age)                  { "adult"  }
multi age-group(Person $person) { age-group $person.age }

这对于每个 Person 例子都有效, 还有它们的 ages.

现在,让我们使用 age-group 定义一个叫做 print-namemulti 来分发.
根据 age-group 分发最明显的方法是使用 where 从句.

multi print-name(Person $person where (age-group($person) eq "child")) { "Little {$person.name}" }
multi print-name(Person $person where (age-group($person) eq "adult")) { $person.name            }
multi print-name(Person $person where (age-group($person) eq "senior")){ "Old Man {$person.name}"}

双引号字符串中的 {$person.name} 将 block 的结果插值到字符串中.

让我们再试试:

> print-name Person.new(:name<timmy>, :age(10))
Little Timmy
> print-name Person.new(:name<john>, :age(23))
John
> print-name Person.new(:name<ezekiel>, :age(89))
Old Man Ezekiel

那很棒. 但是如果我们有更多的基于 person 的 age-group 的 multis 要分发呢? 难道我们真的每次都要写出 (Person $person where (age-group($person) eq "child")) 这样的代码吗? 不, 我们不需要, 感谢 subset 类型.

subset Child  of Person where *.age < 16;
subset Adult  of Person where -> $person { 16 <= $person.age < 66 };
subset Senior of Person where *.age >= 66;

multi print-name(Child $person)  { "Little {$person.name}"  }
multi print-name(Adult $person)  { $person.name             }
multi print-name(Senior $person) { "Old Man {$person.name}" }

由于 Rakudo 在处理含有组合的链式比较操作符的 Whatever 时有一个 bug, 我们不得不为 Adult 写一个显式的 block.

这个 bug 现已修复, 所以:

subset Adult  of Person where -> $person { 16 <= $person.age < 66 };

等价于:

subset Adult  of Person where  16 <= *.age < 66;

这个新版本的 print-name 与之前旧版本产生同样的结果. 现在我们能从 Child/Adult/Senior 的角度重写 age-group :

multi age-group(Child)  { "child"  }
multi age-group(Adult)  { "adult"  }
multi age-group(Senior) { "senior" }
multi age-group(Int $age) { age-group Person.new(:$age) }

:$age:age($age) 的简写方式.

又一次, 我们有了产生想要的结果的更清晰的代码, 多亏了 multiple 分发和 subset 类型.