Perl 6 - 适当的使用 proto

作者: 焉知非鱼 | 来源:发表于2016-03-11 03:19 被阅读24次

    原文在此Apropos proto: Perl6.c multi thoughts

    Multi 程序相当整洁, 但对于我来说是不彻底的。一些背景 — 有人可能这样计算阶乘:

    multi fac(0) { 1 }
    multi fac(Int $n where 1..Inf) { $n * fac( $n-1 ) }
    say fac(4); # 24
    

    现在假设我们要把我们的递归 multi-sub 作为一个回调传递会怎样呢?

    given &fac -> $some_fun { say "some_fun(4)=", $some_fun(4) }
    

    现在... 定义一个匿名的 multi-sub 怎么样?

    my $anon_fac = do {
        multi hidden_fac(0) { 1 }
        multi hidden_fac(Int $n where 1..Inf) { $n * fac( $n - 1 ) }
        &hidden_fac };
       
    say $anon_fac(4); # 24
    

    这也会有作用, 但是有点 hack 的味道, 并且我们的 multi-sub 并不是真正的匿名。它仅仅是被隐藏了。真正匿名的对象不会在任何作用域中安装, 而在这个例子中, "hidden_fac" 被安装在 "do" block 中的本地作用域中。

    Perl 6说明书没有排除匿名的 multi 程序, 而且事实上

    my $anon_fac = anon multi sub(0) { 1 }
    

    会报一个错误:

    Cannot use 'anon' with individual multi candidates. Please declare an anon-scoped proto instead

    不能对单独的 multi 候选者使用 anon。请声明一个 anon-scoped 的 proto 代替。

    让我们回到原先那个以 "multi fac(0) { 1 }" 开始的例子。当编译器看到它, 就会在同一个作用域中为我们创建一个"proto fac" 作为 multi 定义。proto 的作用就像一个分发器(dispatcher) — 从概念上讲, 当我们调用 fac(4) 的时候, 我们让 proto fac 为我们从 multi facs 中挑选一个出来以调用。

    我们可以提前显式地定义一个 proto, 而且我们甚至能通过指定它的所有程序都需要 Int 类型的参数来对默认的 "proto" 加以改良。

    proto fac_with_proto(Int) { * }
    multi fac_with_proto(0)   { 1 }
    multi fac_with_proto(Int $n where 1..Inf) { $n * fac( $n - 1 ) }
    say fac_with_proto(4); # 24
    

    因此, anon muiti sub 抛出的错误 — Please declare an anon-scoped proto instead — 正是告诉我们 "没有要安装到的作用域, 我不能为你获取一个 proto。 使用你自己的 anon proto, 并把这个程序附加给它"。

    好的, 花蝴蝶, 感谢你的提醒! 我试试...

    my $fac_proto = anon proto uninstalled-fac(Int) { * };
    say $fac_proto.name; # uninstalled-fac
    

    好极了! 现在所有我们要做的就是给那个 proto 添加 multis。

    $fac_proto 是一个 Sub 对象, 它有方法来告诉你候选者, 但是没有办法设置(set) 候选者。并且我找不到任何方式在创建时传递一个候选者列表。

    适当的修补


    什么会让 proto/multi 干净并且正交是一种方式去

    • 在编译时指定候选者
    • 在运行时添加候选者

    这有点像

    my $future_fac = Proto( :dispatch( sub (Int) {*} ),
                            :candidates( [sub (0) {1}] ),
                            :mutable );
    
    $future_fac.candidates.push(
        sub (Int $n where 1..Inf) { $n * fac( $n-1 ) }
      );
    
    $future_fac(4); # 24
    

    我假定了一个 Sub 的子类 Proto 以揭露 multi 程序的内部工作原理。这个构造函数会允许定义任何 proto 声明符所做的: 签名 & 默认程序和名字。 还有, 它会允许在初始的候选者列表中传递一个属性。

    最后, 那个对象自身会让候选者方法返回一个数组, 而不是一个不可变列表, 如果 Proto 是使用 mutable 属性创建的话。不指定 mutable 将意味着所有的 multis 需要在编译时添加, 而不允许在运行时添加。

    相关文章

      网友评论

        本文标题:Perl 6 - 适当的使用 proto

        本文链接:https://www.haomeiwen.com/subject/zwntlttx.html