NAME
BEGIN::Lift - Lift subroutine calls into the BEGIN phase
SYNOPSIS
package Cariboo;
use strict;
use warnings;
use BEGIN::Lift;
sub import {
my $caller = caller;
BEGIN::Lift::install(
($caller, 'extends') => sub {
no strict 'refs';
@{$caller . '::ISA'} = @_;
}
);
}
package Foo;
use Cariboo;
extends 'Bar';
# functionally equivalent to ...
# BEGIN { @ISA = ('Bar') }
DESCRIPTION
This module serves a very specific purpose, which is to provide a mechanism through which we can "lift" a given subroutine to be executed entirely within the BEGIN
phase of the Perl compiler and to leave no trace of itself in the RUN
phase.
Modules loaded at runtime?
If a package that uses this module is loaded at runtime (perhaps via the require
builtin), it will still work correctly (to the best of my knowledge that is).
FUNCTIONS
install( $package, $keyword_name, $keyword_handler )
This will install a lifted subroutine named $keyword_name
into the specified $package
. All calls to this the lifted subroutine will execute the $keyword_handler
immediately after parsing it.
If this subroutine is called outside of the BEGIN
phase, an error will be thrown. If there already exists a typeglob for $keyword_name
then an error will be thrown.
CAVEATS
All arguments to lifted subroutines must be BEGIN
time safe
This means they require no runtime initiatlization or access to runtime initialized variables (as they won't be initialized).
For instance, given a lifted subroutine called add
, this code is BEGIN
time safe because the arguments are numeric literals.
add( 1, 1 );
While this version is not safe because it relies on the @args
variable being initialized at runtime.
my @args = (1, 1);
add( @args );
PARSING ISSUES
Ideally we can (eventually) detect these situations and error accordingly so that this is no longer a burden to the user of this module, but instead just part of the normal operation of it.
Non-void context
If, for instance, a lifted sub is called such that the return value is to be assigned to a variable, such as:
my $x = my_lifted_sub();
It will not behave as expected, since my_lifted_sub
is evaluated entirely at BEGIN
time, the resulting value for $x
at RUN
time is undef
.
Expression context
If, for instance, a lifted sub is called within an expression where the return value is important, such as:
if ( my_lifted_sub() && 10 ) { ... }
It will not behave as expected, since my_lifted_sub
is evaluated entirely at BEGIN
time and has the value of undef
at runtime, the conditional will always fail.
Statement modifier context
If, for instance, a lifted sub call is guarded by a statement modifier, such as:
my_lifted_sub() if 0;
It will not behave as expected, since the lifted sub call is evaluated entirely at BEGIN
time the statement modifier has no affect at all and <my_lifted_sub> will always be executed.
SEE ALSO
- Devel::BeginLift
-
This does a similar thing, but does it via "some slightly insane perlguts magic", while this module has much the same goals, it will (hopefully) accomplish it with less insanity.
AUTHOR
Stevan Little <stevan@cpan.org>
COPYRIGHT AND LICENSE
This software is copyright (c) 2017 by Stevan Little.
This is free software; you can redistribute it and/or modify it under the same terms as the Perl 5 programming language system itself.