12

考虑这个例子,其中一个子类有一个没有签名的 multi 方法和一个带有 slurpy 参数的方法:

class Foo {
    multi method do-it { put "Default" }
    multi method do-it ( Int $n ) { put "Int method" }
    multi method do-it ( Str $s ) { put "Str method" }
    multi method do-it ( Rat $r ) { put "Rat method" }
    }

class Bar is Foo {
    multi method do-it { put "Bar method" }
    multi method do-it (*@a) { put "Bar slurpy method" }
    }

Foo.new.do-it: 1;
Foo.new.do-it: 'Perl 6';
Foo.new.do-it: <1/137>;
Foo.new.do-it;

put '-' x 10;

Bar.new.do-it: 1;
Bar.new.do-it: 'Perl 6';
Bar.new.do-it: <1/137>;
Bar.new.do-it: 5+3i;
Bar.new.do-it;

方法查找的结构如何?我正在寻找更多解释它的方法,特别是不抱怨它。

Int method
Str method
Rat method
Default
----------
Int method
Str method
Rat method
Bar slurpy method
Bar method

Bar例如,有一个对'sdo-it的调用1。一些理性的人可能会认为它首先会寻找匹配的签名,Bar并且 slurpy 永远不会让任何东西通过它。然而,调用在继承链中找到了正确的 multi。

是否Bar已经知道所有签名?它是搜索还是在编写时所有这些东西都已经解决了?

而且,有没有办法在运行时找出哪个类提供了该方法?也许有一些关于如何的电话?当我有一个我错误地指定并且正在其他地方处理的 multi 时,这将是一个方便的调试工具。

4

2 回答 2

13

多重分派要记住的关键是它发生子或方法解析发生之后。所以所有的多次调度实际上是一个两步的过程。这两个步骤也是相互独立的。

在写类似的东西时:

multi sub foo($x) { }
multi sub foo($x, $y) { }

编译器将生成:

proto sub foo(|) {*}

也就是说,除非你proto自己写了一个子。这proto是实际安装到 lexpad 中的内容;sub 永远不会直接安装到multilexpad 中,而是安装到proto.

所以,调用multisub的时候,流程是:

  1. 使用词法查找找到要调用的子程序,它解析为proto
  2. 调用proto,它选择最佳multi候选人并调用它

multi当嵌套作用域中有候选者时,proto来自外部作用域的将被克隆并安装到内部作用域中,并将候选者添加到克隆中。

多种方法会发生非常相似的过程,除了:

  • 多个方法只是存储在一个待办事项列表中,直到}类、角色或语法关闭
  • Aproto可能由角色或班级提供,因此与multi候选人组成角色只需将他们也添加到待办事项列表中
  • 最后,如果有多个方法没有proto,但父类有这样的proto,那将被克隆;proto否则会做空

这意味着对多方法的调用是:

  1. 使用通常的方法分派算法(仅使用 C3 方法解析顺序搜索类)查找方法,该算法解析为proto
  2. 调用proto,它选择最佳multi候选人并调用它

多子和多方法都使用完全相同的排序和选择算法。就多重分派算法而言,调用者只是第一个参数。此外,Perl 6 多重分派算法不会比后面的参数更重地加权较早的参数,因此就像:

class A { }
class B is A { }
multi sub f(A, B) { }
multi sub f(B, A) { }

会被认为是捆绑的,如果用 调用f(B, B)会给出一个模棱两可的调度错误,所以定义:

class B { ... }
class A {
    multi method m(B) { }
}
class B is A {
    multi method m(A) { }
}

然后调用B.m(B),因为 multi-dipsatcher 再次只看到类型元组(A, B)(B, A)

多重调度本身关注的是窄的概念。如果 C1 的至少一个参数的类型比 C2 中相同位置的参数的类型更窄,并且所有其他参数都被绑定(即不窄,不宽),则候选 C1 比 C2 窄。如果反之为真,则它更宽。否则,它是绑定的。一些例子:

(Int) is narrower than (Any)
(Int) is tied with (Num)
(Int) is tied with (Int)
(Int, Int) is narrower than (Any, Any)
(Any, Int) is narrower than (Any, Any)
(Int, Any) is narrower than (Any, Any)
(Int, Int) is narrower than (Int, Any)
(Int, Int) is narrower than (Any, Int)
(Int, Any) is tied with (Any, Int)
(Int, Int) is tied with (Int, Int)

multi-dipsatcher 构建候选者的有向图,其中只要 C1 比 C2 窄,就会有一条从 C1 到 C2 的边。然后它找到所有没有传入边的候选,并将它们删除。这是第一批候选人。删除将产生一组没有传入边的新候选,然后将其删除并成为第二组候选。这种情况一直持续到所有候选者都从图中取出,或者如果我们达到无法从图中取出任何内容的状态(非常罕见的情况,但这将作为循环报告给程序员)。这个过程发生一次,而不是每次调度,它会产生一组候选组。(是的,这只是一种拓扑排序,但分组细节对于接下来的内容很重要。)

发生呼叫时,将搜索组以查找匹配的候选人。如果同一组中的两个候选人匹配,并且没有决胜局(命名参数、子句或来自类型、解包或的where隐含子句),则将报告模棱两可的调度。如果搜索所有组而没有找到结果,则分派失败。wheresubsetis default

关于 arity 也有一些狭隘的考虑(必需的参数比可选参数或 slurpy)和is rw(它比没有 的其他相等的候选者更窄is rw)。

一旦发现一组中的一个或多个候选人匹配,则考虑决胜局。这些包括命名参数、where子句和解包的存在,并在首场比赛获胜的基础上工作。

multi f($i where $i < 3) { } # C1
multi f($i where $i > 1) { } # C2
f(2) # C1 and C2 tied; C1 wins by textual ordering due to where

请注意,此文本排序仅适用于平局;就类型而言,源代码中候选的顺序并不重要。(命名参数也仅充当决胜局,有时令人惊讶。)

最后,我要指出,虽然多次分派的结果将始终与我所描述的两步过程相匹配,但实际上发生了大量的运行时优化。虽然所有查找最初都完全按照描述进行解析,但结果被放入调度缓存中,它提供的查找速度比搜索拓扑排序提供的组快得多。这是以这样一种方式安装的,即可以完全绕过 proto 的调用,从而节省调用帧。如果您 ; 您可以看到此行为的伪影--profile;与多个候选者相比,为任何基于类型的调度(没有决胜局)自动生成proto的调用将收到少量调用。当然,如果您在 中编写自定义逻辑,这将不适用proto

除此之外,如果你在 MoarVM 上运行,动态优化器可以走得更远。它可以使用收集和推断的类型信息来解决方法/子分派多分派,将2步过程变成0步过程。小的候选人也可以内联到调用者中(同样,分析器可以告诉您内联已经发生),这可以说将多调度变成了 -1 步过程。:-)

于 2017-07-15T23:52:24.840 回答
6

Rakudo Perl 6 方法查找过程默认由Metamodel::MROBasedMethodDispatch 角色完成。请参阅Rakudo 的 /src/Perl6/Metamodel/MROBasedMethodDispatch.nqp以获取相应的源代码。

(默认情况下,它使用角色 Metamodel::C3MRO,它实现了C3 方法解析顺序。请参阅Rakudo 的 /src/Perl6/Metamodel/C3MRO.nqp获取源代码。)

.^find_method返回基于短名称(不带参数)的匹配方法。只要短名称对应于多个方法,这个返回的方法就是一个proto

调用.candidatesproto 会返回与 proto 匹配的Method 对象列表。(调用.candidates非原型方法只会返回与单元素列表中唯一元素相同的方法。)

for Bar.^find_method('do-it').candidates -> $method {
    $method.signature.say;
}

这使:

(Foo $: *%_)
(Foo $: Int $n, *%_)
(Foo $: Str $s, *%_)
(Foo $: Rat $r, *%_)
(Bar $: *%_)
(Bar $: *@a, *%_)

Bar.new.do-it: 5+3i;调用传递一个Barasself加上5+3i位置参数。候选列表中最接近这些参数的签名(又名“最窄匹配”)就是其中(Bar $: *@a, *%_)一个。因此,具有该签名的例程被调用。

Bar.new.do-it;调用传递了一个Baras而self没有别的。(Bar $: *%_)签名是比 更接近(更窄)的匹配(Bar $: *@a, *%_)。同样,具有最接近(最窄)签名的例程被调用。

于 2017-07-12T08:32:43.553 回答