开发者

Why can't I use a Perl variable's value to access a lexical variable name?

开发者 https://www.devze.com 2022-12-21 02:48 出处:网络
Why does this print 42: $answer = 42; $variable = \"answer\"; 开发者_运维知识库 print ${$variable} . \"\\n\";

Why does this print 42:

$answer = 42;
$variable = "answer";
开发者_运维知识库
print ${$variable} . "\n";

but this doesn't:

my $answer = 42;
my $variable = "answer";

print ${$variable} . "\n";


Only package variables (the kind declared in your first example) can be targeted via symbolic references. Lexical (my) variables, cannot be, which is why your second example fails.

See the excellent article Coping with Scoping for how the two separate variable systems in Perl operate. And see the also excellent Why it's stupid to use a variable variable name for why that's stupid. :)


Perl has two entirely separate but largely compatible variable systems, package variables, as in your first example, and lexical variables, as in the second. There are a few things that each can do but the other cant:

Package variables are the only ones that can be:

  1. localized (with local)
  2. used as the target for a symbolic reference (the reason the OP's second example doesnt work)
  3. used as barewords (sub definitions, file handles)
  4. used with typeglobs (because that's what the symbol really is under the hood)

Lexical variables are the only ones that can be closed over (used in a lexical closure).

Using strict would help by forcing you to declare package variables with our, making the difference clearer.

There are several times where symbolic references are useful in Perl, most center around manipulating the symbol table (like writing your own import in a module rather than using Exporter, monkey-patching modules at runtime, various other meta-programming tasks). All of these are advanced topics.

For other tasks, there is usually a better way to do it such as with a hash. The general rule of thumb is to always run under use warnings; use strict; unless you know there isn't any other way but to disable a portion of the pragma (such as using no strict 'refs'; in as small a scope as possible).


Symbolic references only work with package variables. The symbol table doesn't track lexical variables (which is the entire point of them being lexical :).


The problem is that you can't use a symbolic reference to refer to a lexical variable. In both examples ${$variable} is looking for $main::answer. In the first, $answer is a package global and short for $main::answer, so the reference finds it. In the second, $answer is a lexical variable and doesn't live in any package, so the reference can't find it.

More details in perlref under heading Symbolic references.

0

精彩评论

暂无评论...
验证码 换一张
取 消