如何修改传递给子例程引用的标量引用?

Mar*_*ark 4 perl pass-by-reference subroutine

我有一个函数将文档转换为不同的格式,然后根据类型文档调用另一个函数.除了需要进行一些清理的HTML文档之外,所有内容都非常简单,而且根据它来自何处进行清理会有所不同.所以我有一个想法,我可以将一个子程序的引用传递给转换函数,这样调用者就有机会修改HTML,有点像(我不在工作,所以这不是复制和粘贴) :

package Converter;
...
sub convert
{
    my ($self, $filename, $coderef) = @_;

    if ($filename =~ /html?$/i) {
        $self->_convert_html($filename, $coderef);
    }
}

sub _convert_html
{
    my ($self, $filename, $coderef) = @_;

    my $html = $self->slurp($filename);
    $coderef->(\$html); #this modifies the html
    $self->save_to_file($filename, $html);
}
Run Code Online (Sandbox Code Playgroud)

然后通过以下方式调用:

Converter->new->convert("./whatever.html", sub { s/<html>/<xml>/i });
Run Code Online (Sandbox Code Playgroud)

我沿着这些方向尝试了几个不同的东西,但我继续得到'在替换中使用未初始化的值(s ///)'.有什么方法可以做我想做的事情吗?

谢谢

dao*_*oad 5

如果是我,我会避免修改标量引用并返回更改后的值:

sub _convert_html
{
    my ($self, $filename, $coderef) = @_;

    my $html = $self->slurp($filename);
    $html = $coderef->( $html ); #this modifies the html
    $self->save_to_file($filename, $html);
}
Run Code Online (Sandbox Code Playgroud)

但是,如果要修改sub的参数,则值得知道所有子参数都是Perl中的pass-by-reference(@_别名为sub调用的参数).所以你的转换子看起来像:

sub { $_[0] =~ s/<html>/<xml>/ }
Run Code Online (Sandbox Code Playgroud)

但是,如果您真的想要操作$_,就像您在所需的代码示例中一样,您需要_convert_html()看起来像:

sub _convert_html
{
    my ($self, $filename, $coderef) = @_;

    my $html = $self->slurp($filename);

    $coderef->() for $html;

    $self->save_to_file($filename, $html);
}
Run Code Online (Sandbox Code Playgroud)

for是一种正确本地化的简单方法$_.你也可以这样做:

sub _convert_html
{
    my ($self, $filename, $coderef) = @_;

    local $_ = $self->slurp($filename);

    $coderef->();

    $self->save_to_file($filename, $_);
}
Run Code Online (Sandbox Code Playgroud)

  • 这是很多复制:首先进入参数堆栈,然后更改它,然后将其复制回参数堆栈.如果你必须处理许多页面,这可能会受到伤害. (2认同)