I have a set of constants declared in Perl:
use constant C1 => 111;
use constant C2 => 222;
..
use constant C9 => 999;
my $which_constant = "C2";
How do I construct a Perl expression which, based on $which_constant
, derives the value of a constant named with the value of this variable - e.g. "222".
Please note that I can not change any of the conditions above - they are a simplification of a real scenario: I have a module (which I have no control over) from which these constants are imported. The `name of one of the constants is supplied by the user from command line. I need to access the appropriate constants' value.
I've been beating my head against the wall (mostly around all sorts of weird glob constructs) but none of them work.
P.S. If the solution accesses the constants inside their native module - say My::Constants::C2
(without needing to import them), even better, but not necessary - I can import the correct constants into main::
easily using My::Constants->import($which_constant)
. and yes, to top it off, te constants are NOT exported by default thus needing the explicit import() call.
Some of the things I tried:
main::$which_constant
- syntax errormain::${which_constant}
- syntax error${*$which_constant}
- returns empty value*$which_constant
- returns "*main::C2"${*${*which_constant}}
- empty
Constants defined by
constant.pm
are just subroutines. You can use method invocation syntax if you have the name of the constant in a string:This way, if you try to use a constant which is not defined, you will get an error.
Sinan's suggestion to use method invocation semantics to get around
strict 'refs'
limits is the cleanest, easiest to read solution.My only concern about this was that the speed penalty for using this approach might be a problem. We've all heard about method call performance penalties and the speed benefits of inlineable functions.
So I decided to run a benchmark (code and results follow).
The results show that normal, inlined constants run about twice as fast as method calls with a literal subroutine name, and almost three times as fast as method calls with variable subroutine names. The slowest approach is a standard deref and invocation of
no strict "refs";
.But, even the slowest approach is pretty darn fast at over 1.4 million times a second on my system.
These benchmarks obliterate my one reservation about using the method call approach to solve this problem.
And the results:
Results generated with ActivePerl 826 on Windows XP, YMMV.
Perl "constants" are actually subroutines that return a constant value. The perl compiler is able to replace them with the appropriate value at compile time. However, since you want to get the value based on a runtime name lookup, you should do:
(And of course you need
no strict 'refs'
somewhere.)