Il debugger di Perl può scavare nel modo desiderato. Per esempio:
main::(-e:1): 0
DB<1> sub foo {}
DB<2> x \&foo
0 CODE(0xca6898)
-> &main::foo in (eval 5)[/usr/share/perl/5.10/perl5db.pl:638]:2-2
Lo fa utilizzando Devel::Peek:
=head2 C<CvGV_name_or_bust> I<coderef>
Calls L<Devel::Peek> to try to find the glob the ref lives in; returns
C<undef> if L<Devel::Peek> can't be loaded, or if C<Devel::Peek::CvGV> can't
find a glob for this ref.
Returns C<< I<package>::I<glob name> >> if the code ref is found in a glob.
=cut
sub CvGV_name_or_bust {
my $in = shift;
return unless ref $in;
$in = \&$in; # Hard reference...
eval { require Devel::Peek; 1 } or return;
my $gv = Devel::Peek::CvGV($in) or return;
*$gv{PACKAGE} . '::' . *$gv{NAME};
} ## end sub CvGV_name_or_bust
si potrebbe esercitare con
#! /usr/bin/perl
use warnings;
use strict;
package Foo;
sub bar {}
package main;
BEGIN { *baz = \&Foo::bar }
sub CvGV_name_or_bust { ... }
print CvGV_name_or_bust(\&baz), "\n";
uscita:
Foo::bar
Si noti che l'esempio precedente dà Foo:bar
un nome diverso, ma si ottiene sia il pacchetto in cui si trova il sub subordinato con alias sia il suo nome.
fonte
2010-09-10 14:13:01
Quale problema stai cercando di risolvere conoscendo questo? Potrebbe esserci un modo per sistemarlo prima che accada. :) –
La modifica di funzioni da altre classi può essere comune in Ruby, ma è una tecnica nera in Perl che dovrebbe essere riservata per * estremamente * rare situazioni. Perché hai bisogno di fare questo? – Ether
@ 'brian d foy' Tre anni dopo sto provando a scrivere un test unitario per confermare che ottengo correttamente una versione di Perl pura di una funzione in una condizione, o una versione XS in un'altra condizione, e all'improvviso ho quello che sento è un caso d'uso legittimo per il quesito di questo PO. ;) – DavidO