5

我们有一个用于执行维护的 Perl 脚本。我需要对其进行修改以处理更多任务。问题是脚本已编译,并且源代码很久以前就丢失了。

我尝试使用 B::Deparse 来重新创建文件,但 Deparse 并不完美,并且输出被破坏(以及非常大的 ~5000 行 deparsed 代码)。

在阅读了解析后的代码后,我发现我需要修改一个函数。编译后的脚本会加载一个纯文本脚本模块,因此我更改了模块以覆盖该函数并执行我需要它执行的任务。现在的问题是我无法访问主脚本“我的”变量。

这是一个例子:

# main.pl

my $a = 1;

sub call_me {
    print "unmodified";
}

use MOD;

call_me;


MOD.pm
package MOD;

main::{'call_me'} = sub {
    print "\$main::a = $main::a\n";
}

结果是:“ $main::a =”而不是获得真正的价值。

提前致谢。

4

2 回答 2

8

简短的回答是声明的变量my在其词法范围之外是不可访问的。如果您不能将声明更改为“我们的”(由于原始脚本的疯狂“编译”性质),那么您还没有倒霉。Perl 几乎总是提供一种方法来解决这些类型的事情。

在这种情况下,您可以安装PadWalker模块并执行以下操作(这是您最初发布的代码的调整版本):

main.pl脚本:

my $a = 1;

sub call_me {
    print "unmodified: $a";
}

use MOD;

call_me;

然后你的模块:

package MOD;

# closed_over($code_ref) returns a hash ref keyed on variable
# name(including sigil) with values as references to the value
# of those variables
use PadWalker qw(closed_over);

{
    # grab a reference to the original sub
    my $orig = \&main::call_me;

    # no need to use the symbol table, a glob reference is fine
    # but you can't use sub main::call_me { ... } either
    *main::call_me = sub {
        my $a = closed_over($orig)->{'$a'};
        print "\$main::a = $$a\n";
    }
}
于 2012-07-13T20:51:17.753 回答
0

如果您有想要的特定内容,这是一个单行字。根据需要扩展。

perl -e '$foo = eval ( cat ./my.pl." return \$some_my_var;"); 打印“$foo”'

于 2018-07-18T23:35:46.577 回答