构建一个perl模块

时间:2011-08-15 22:25:07

标签: perl design-patterns

我有一个.pm模块,它有五个函数,每个函数返回0或1,我想添加另一个函数,在将来reouln 0或1(这些函数是测试函数1表示ok 0表示失败)。

我想从脚本中调用那些基于.pm模块的函数。

我希望我的脚本调用.pm模块上的每个函数,如果它返回0然后继续到下一个函数。如果它返回1,那么它应该将一些东西打印到LOG并停止处理该记录。

假设我将更新.pm模块并向其添加新功能,是否可以保留脚本代码而不做任何更改?我每次将测试添加到.pm模块时都不想添加if条件?

1 个答案:

答案 0 :(得分:3)

.pm模块应该提供一种检索要测试的函数列表的方法。

我认为最好的方法是它是一个子程序调用,但你也可以使用包中定义的变量(例如列表变量)。

示例:

package MyModule;

sub testable_functions { qw(fun1 fun2 fun3) }
sub fun1 { ... }
sub fun2 { ... }
sub fun3 { ... }
sub not_going_to_be_tested { ... }

或:

package MyModule;
our @testable_functions = qw(fun1 fun2 fun3);

在您的测试代码中:

my @funs_to_be_tested = MyModule->testable_functions;
# or = @MyModule::testable_functions if you're using a list

for my $fun (@funs_to_be_tested) {
  my $full_name = "MyModule::" . $fun;
  $full_name->() or die "function $full_name failed\n";
}

现在,您可以在不更改测试代码的情况下添加要测试的功能。

如果你想获得幻想,你可以翻阅包装的符号表:

package MyModule;

sub testable_functions {
  my @funs;
  for my $name ( keys %MyModule:: ) {
    next if $name eq "testable_functions"; # can add more conditions here
    my $full_name = "MyModule::".$name;
    next unless *{$full_name}{CODE};       # avoid non-subs
    push(@funs, $name);
  }
  return @funs;
}

尽管如此,合同是一样的:MyModule为测试代码提供了一种获取应该测试的函数列表的方法。