4

如果文件已经加载,是否有任何方法可以连接到文件中,use/require以便我可以抛出异常?在我即将到来nextgen::blacklist的 中,如果使用某些模块,我会尝试死亡。我正在使用中提到的对象挂钩方法perldoc -f require:有三类挂钩object带有 subref 的数组subref。这篇文章中的例子是 object-hook,你可以在nextgen::blacklist.

我想要的语法是这样的:

perl -Mnextgen -E"use NEXT"

package Foo;
use nextgen;
use NEXT;

理想情况下,它应该抛出这样的消息:

nextgen::blacklist violation with import attempt for: [ NEXT (NEXT.pm) ] try 'use mro' instead.

我已经尝试了很多不同的方法。

package Class;
use Data::Dumper;
use strict;
use warnings;

sub install {
  unshift @main::INC, bless {}, __PACKAGE__
    unless ref $main::INC[0] eq __PACKAGE__
  ;
}

sub reset_cache { undef %main::INC }

sub Class::INC {
  my ( $self, $pmfile ) = @_;
  warn Dumper [\%main::INC, $pmfile];
  #undef %INC;
} 

package main;
  BEGIN { Class->install; undef %main::INC }
  use strict;
  use strict;
  use strict;
  use strict;
  use warnings;
  use strict;
  use warnings;

似乎%INC这些钩子之后设置。我对任何可以让我抛出异常的东西都感兴趣。如果尝试加载/重新加载一个模块,尽管它的状态是其他不使用我的编译指示的模块的依赖项,我想死。

package Foo;
use NEXT;

package main;
use Foo; (which uses Next.pm);
use NEXT.pm; ## Throw exception
4

1 回答 1

5

您可能希望将 coderef 放在开头的 @INC 上,如perldoc -f require. 从那里,您可以引发异常以防止某些模块被加载,或者什么都不做以让 require 继续其在其他 @INC 条目中查找模块的正常工作。

$ perl -E'BEGIN { unshift @INC, sub { die q{no NEXT} if pop eq q{NEXT.pm}; () }; }; use Carp; say q{success}'
success
$ perl -E'BEGIN { unshift @INC, sub { die q{no NEXT} if pop eq q{NEXT.pm}; () }; }; use NEXT; say q{success}'
no NEXT at -e line 1.
BEGIN failed--compilation aborted at -e line 1.

如果你希望这种行为是词法的,你应该使用 Perl 的提示 hash %^H。处理这个有点繁琐,所以我建议使用Devel::Pragma,它可以为你处理所有血腥的细节。

正如您所指出的,@INC不会为已加载的模块执行挂钩。如果您还需要挂钩到userequire加载的模块,则覆盖CORE::GLOBAL::require将起作用,因为每次尝试加载模块时都会调用它。

$ perl -E'BEGIN { *CORE::GLOBAL::require = sub { warn @_ } } use NEXT; use NEXT;'
NEXT.pm at -e line 1
NEXT.pm at -e line 1.

另外,作为 NEXT 的维护者,我完全赞成永远阻止人们使用它。:-)

于 2010-10-06T02:15:04.173 回答