如何指定包含我接受的所有内容的Perl 6签名并排除其他所有内容?

前端之家收集整理的这篇文章主要介绍了如何指定包含我接受的所有内容的Perl 6签名并排除其他所有内容?前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
我在这个问题中的假设是,我指定的签名完全描述了方法将接受的所有内容.显然我错了,但无论如何我想要那样做.如果我没有在签名中指定它,我不希望其他人能够提供它作为一个认为它会做某事的论据.

我为方法创建了一个带有单个位置参数的签名,我希望它只接受单个位置参数.但是,它也接受命名参数而无需投诉:

class SomeClass {
    method something ( Int $n ) { 'Ran something' }
    }

put SomeClass.something: 137; # Ran something
put SomeClass.something: 137,:foo('bar'); # Ran something

但是,如果我定义一个采用命名参数的方法,那么每次我定义它时都会调用它.此外,虽然我认为我已经说过它需要一个名为paramter foo,但它不是必需的,它仍然接受我没有指定的命名参数:

class SomeClass {
    multi method something ( Int $n ) { 'Ran something' }
    multi method something ( Int $n,:$foo ) { "Ran $foo" }
    }

put SomeClass.something: 137;  # Ran
put SomeClass.something: 137,:foo('bar'); # Ran bar
put SomeClass.something: 137,:bar('foo'); # Ran

所以,有些问题:

>如何指定包含我想接受的所有内容的签名并排除其他所有内容
>如何强制Perl 6选择最接近的匹配签名?
> Perl 6以什么顺序决定检查方法

解决方法

您正在谈论有关多方法调度最难掌握的事情之一.最重要的是要实现每个方法签名都有一个隐含的*%_(也就是一个slurpy哈希),如果没有指定的话.这意味着它将吃掉任何非特定的命名参数.
class A {
    method a() { dd %_ }  # %_ exists even if not specifically specified
}
A.a(:foo)   # {:foo}

您需要意识到的第二件事是,命名参数只能起到打破平局的作用.因此,如果有多个候选者具有相同的位置参数匹配集,MMD将使用第一个有效的候选者(考虑到所有意外的命名参数都被%_ sl):

class A {
    multi method a(:$foo) { say "foo" }
    multi method a(:$bar) { say "bar" }
}
A.a(:bar)    # foo

这个看似意外的结果是由以下事实引起的:

>两个候选人具有相同数量的位置参数
>第一个候选匹配因为:$foo是可选的
>和:bar被隐含的*%_吃掉

为了使它更像您期望的工作,您需要将需要打破的候选项放入您希望它们触发的顺序,并使任何命名参数成为必需参数:

class A {
    multi method a(:$foo!) { say "foo" }
    multi method a(:$bar!) { say "bar" }
}
A.a(:bar)    # bar

如果您有候选者采用多个命名参数,它很快就会变得相当复杂,只要有一个使用%_的内省的方法,你可能会更好:

class A {
    method a() {
        if %_<foo> {
            say "foo"
        }
        elsif %_<bar> {
            say "bar"
        }
        else {
            die "None of the signatures matched"
        }
    }
}
A.a(:bar)    # bar

希望这会让事情更加清晰:-)

猜你在找的Perl相关文章