perl - 使用 Sub::Quote 弱化捕获

标签 perl moo

我想削弱 Sub::Quote 生成的代码中捕获的变量.例如,这是未引用的替代方案:

use 5.10.0;
use Scalar::Util qw[ weaken ];
{
  my $s = 'foo';
  my $x = sub { say $s };
  weaken( my $y = $x );

  my $bar = sub { &$y };
  &$bar;
  $x = undef;
  &$bar
}

和输出:
foo
Can't use an undefined value as a subroutine reference [...]

这是我的 Sub::Quote 尝试:
use 5.10.0;
use Sub::Quote;
use Scalar::Util qw[ weaken ];
{
  my $s = 'foo';
  my $x = sub { say $s };
  weaken( my $y = $x );

  my $bar = quote_sub( '&$y', { '$y' => \$y } );
  &$bar;
  $x = undef;
  &$bar;
}

和输出:
foo
foo

显然捕获的$y没有被削弱。有没有办法改变生成的代码来削弱捕获的变量?

文档很少,Sub::Quote实现复杂;我相当相信当前代码不可能做到这一点,但我很乐意被证明是错误的。

最佳答案

my $bar = quote_sub( '&$y', { '$y' => \$y } );

大致相同
my $bar = eval(q{ my $y = $y; sub { &$y } });

(它做得更多,但这些位与这个问题无关)。如您所见,这创建了对 sub[1] 的新强引用。

作为一种解决方法,您可以添加一个间接层:
my $bar = eval(q{ my $y_ref = \$y; sub { &{ $$y_ref } } });

这可以通过使用来实现:
my $bar = quote_sub( '&{$$y_ref}', { '$y_ref' => \\$y } );

如果 $y 不会有任何问题创建者 Sub::Quote 是您的 $y 的别名。 .这可以使用 Data::Alias 或 5.22 中引入的实验特性来实现。

这可以使用以下方法来证明:
{
  package Sub::Quote;

  my $sub = sub {
    my ($from, $captures, $indent) = @_;
    join(
      '',
      "use feature qw( refaliasing );\n",
      "no warnings qw( experimental::refaliasing );\n",
      map {
        /^([\@\%\$])/
          or croak "capture key should start with \@, \% or \$: $_";
        (' ' x $indent).qq{\\my ${_} = \\${1}{${from}->{${\quotify $_}}};\n};
      } keys %$captures
    )
  };

  no warnings qw( redefine );
  *capture_unroll = $sub;
}


my $bar = quote_sub( '&$y', { '$y' => \$y } );

您可以与模块的维护者讨论添加一个会导致使用别名的选项。

  • 当您创建(强或弱)引用的副本时,它就是强引用。
  • 关于perl - 使用 Sub::Quote 弱化捕获,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40811097/

    相关文章:

    perl - 当方法是 CODEREF 时,继承如何工作?

    linux - Debian server "Can' t locate loadable object"yet "Module is up to date"

    perl - 如何模拟 Moo 角色中定义的方法?

    perl - 有没有办法覆盖 Moo 中的构造函数?

    python - 执行 python 脚本并读取其 JSON 输出

    perl - 用两个不同的定界符将一条线分成几部分

    珀尔 : How to use constant tag value as key of associative array in?

    perl - 如何在 Moo 中使用 Types::Path::Tiny

    perl - 类型::微小而深刻的强制转换