Perl Moose方法修饰符:在'before'和'after'之前调用'around'

时间:2009-11-09 17:32:14

标签: perl methods moose

我正在使用Moose,我需要在项目中包装方法调用。重要的是我的包装代码是最外层的修饰符。到目前为止我所做的是将我的方法修饰符放在Moose角色中,然后在我的类末尾应用该角色,如下所示:

use Moose::Util;
Moose::Util::apply_all_roles(__PACKAGE__->meta, ('App:Roles::CustomRole'));
__PACKAGE__->meta->make_immutable;

这让我可以合理地确定我的角色的修饰符是最后定义的,因此给了我“之前”和“之后”的正确行为。 (角色中的“之前”和“之后”被称为第一个也是最后一个。)

我原本以为这就足够了,但我现在真的需要用“around”以类似的方式包装方法。构建Moose的Class :: MOP首先应用“around”修饰符,因此它们在“before”之后和“after之后”之后调用。

有关详细信息,请参阅我的修饰符的当前调用顺序:

CUSTOM ROLE before
    before 2
       before 1
           CUSTOM ROLE around
               around
                   method
               around
           CUSTOM ROLE around
       after 1
    after 2
 CUSTOM ROLE AFTER

我真的需要这样的东西:

CUSTOM ROLE before
    CUSTOM ROLE around
        before 2
           before 1
               around
                   method
               around
           after 1
        after 2
    CUSTOM ROLE around
 CUSTOM ROLE AFTER

关于如何在我想要的地方应用/调用我的“around”修饰符的任何想法?我知道我可以做一些符号表黑客攻击(比如Class :: MOP已经在做了),但我真的不愿意。

2 个答案:

答案 0 :(得分:5)

最简单的解决方案是让CUSTOM ROLE定义一个调用main方法然后将其包装的方法。

role MyRole { 
    required 'wrapped_method';
    method custom_role_base_wrapper { $self->wrapped_method(@_) }

    around custom_role_base_wrapper { ... }
    before custom_role_base_wrapper { ... }
}

你遇到的问题是你试图让CUSTOM ROLE包围其他而不是方法。这不是它的目的。除了像你所建议的那样编写类似的符号表hackery(可能你可以争论一个Moose人在Class :: MOP中暴露API以帮助实现),我能想到的唯一其他解决方案就是上面的那个。

如果您不想要custom_role_base_wrapper将添加的额外调用堆栈帧,您应该查看Yuval的Sub::Call::Tail或使用goto来操作调用堆栈。

答案 1 :(得分:3)

我对穆斯很新,但为什么要这样做:

use Moose::Util;
Moose::Util::apply_all_roles(__PACKAGE__->meta, ('App:Roles::CustomRole'));

而不仅仅是这个?

with 'App:Roles::CustomRole';

关于您的问题,这有点像黑客攻击,但您可以将around方法拆分为beforeafter方法,并在类定义结束时应用该角色(所以它按照你想要的顺序应用)?如果绝对必要,您可以使用私有属性在两种方法之间保存状态。