9

我即将为新项目选择使用哪种语言:Perl5 或 Perl6。到目前为止 6 胜,只是缺少Moo的懒惰属性。我在模块中找到的两个实现缺少关键功能。因此,我尝试编写自己的实现。

角色与阶级

我遇到的第一个问题是.package角色中声明的属性的内容。考虑以下内容:

 role HOW1 {
     method compose ( Mu $class ) {
         note "HOW1.compose";
         nextsame;
     }
 }

 role HOW2 {
     method compose ( Mu $class ) {
         note "HOW2.compose";
         nextsame;
     }
 }

 multi trait_mod:<is> (Attribute:D $attr, :$mooish!) {
     note "Attribute's package.HOW: ", $attr.package.HOW;
     note '$*PACKAGE.HOW: ', $*PACKAGE.HOW;
     $attr.package.HOW does HOW1;
     $*PACKAGE.HOW does HOW2;
 }

 class Foo {
     has $.bar is mooish;
 }

 role FooRole {
     has $.baz is mooish;
 }

脚本的输出如下:

Attribute's package.HOW: Perl6::Metamodel::ClassHOW.new
$*PACKAGE.HOW: Perl6::Metamodel::ClassHOW.new
HOW2.compose
HOW1.compose
Attribute's package.HOW: Perl6::Metamodel::GenericHOW.new
$*PACKAGE.HOW: Perl6::Metamodel::ParametricRoleHOW.new
HOW2.compose

从输出中可以清楚地看出,将角色应用于元类始终适用于类并且仅适用于$*PACKAGE.HOW角色。使用 of$*PACKAGE而不是.package可以被认为是一种解决方案,但不是我真正想要使用的解决方案。(虽然,如果没有更好的方法......

存取器

我也想为私有属性提供惰性功能。是的,这self!bar仅适用于语法,但这是我愿意做出的牺牲。问题是到目前为止我发现的所有定制访问器的例子都使用Attribute.set_value()了太低级的方法。我想要这样的东西:

 role MooishHOW {
     method compose ( Mu $class ) {
         my $accessor = $class.^add_private_method( 'bar1',
             method () is rw {
                 note self.WHO, ".bar1";
                 Proxy.new(
                     FETCH => -> $o {
                         $!bar1;
                     },
                     STORE => method ( $val ) {
                         note "Storing";
                         $!bar1 = $val;
                     }
                 );
             }
         );

         callsame;
     }
 }

 multi trait_mod:<is> (Attribute:D $attr, :$mooish!) {
     $attr.package.HOW does MooishHOW unless $attr.package.HOW ~~ MooishHOW;
 }

 class Foo {
     has $.bar is mooish;
     has $!bar1 is mooish;

     method to-bar1 {
         note "bar1 val:",self!bar1;
     }
 }

 my $inst = Foo.new;
 $inst.to-bar1; 

但是由于范围 ( ) $!bar1,符号无法编译。MooishRole是否有我遗漏的技巧可以引用私有属性self

棘手的一个

也许可以使属性成为Proxy容器?这将大大简化惰性实现的整体逻辑。

4

1 回答 1

4

通过最终实现目标并发布AttrX::Mooish模块,我已经回答了我所有的问题。

到目前为止,第一个问题的答案是:没有。$*PACKAGE是目前唯一的办法。

第二个问题:没有答案,但最终的代码set_value()无论如何都要依赖。

棘手的问题碰巧是可能的:set_value()将属性绑定到容器是否可以绑定到Proxy对象。无需牺牲,可以直接访问私有属性,懒惰地处理它们。

谢谢大家,您的回答让我可以解决一些粗糙的问题!

于 2018-08-08T01:46:20.717 回答