2016-10-06 15 views
5

Rozważmy ten mały program Perl, test.pl:Debuger Perla - błąd włamania (wyjątek)?

#!/usr/bin/env perl 
use warnings; 
use strict; 
use Number::Format qw(:subs); # sudo perl -MCPAN -e 'install Number::Format' 

my $tstr = ""; 
my $numFormatter = new Number::Format(); 

for (my $ix=0; $ix<20; $ix++) { 
    $tstr = $tstr . int(rand(10)); 
    my $ftstr = $numFormatter->format_number($tstr, 2, 1); 
    print "ix: $ix ; in: $tstr ; out: $ftstr\n"; 
} 

Gdybym go uruchomić, to nie jest błąd. Jeśli uruchomić go w debugerze Perl, używając perl -d, to również nie powiedzie się z powodu błędu:

$ perl -d test.pl 

Loading DB routines from perl5db.pl version 1.39_10 
Editor support available. 

Enter h or 'h h' for help, or 'man perldebug' for more help. 

main::(test.pl:6): my $tstr = ""; 
    DB<1> c 
ix: 0 ; in: 6 ; out: 6.00 
ix: 1 ; in: 63 ; out: 63.00 
ix: 2 ; in: 637 ; out: 637.00 
ix: 3 ; in: 6379 ; out: 6,379.00 
ix: 4 ; in: 63790 ; out: 63,790.00 
ix: 5 ; in: 637906 ; out: 637,906.00 
ix: 6 ; in: 6379062 ; out: 6,379,062.00 
ix: 7 ; in: 63790624 ; out: 63,790,624.00 
ix: 8 ; in: 637906246 ; out: 637,906,246.00 
ix: 9 ; in: 6379062467 ; out: 6,379,062,467.00 
ix: 10 ; in: 63790624671 ; out: 63,790,624,671.00 
ix: 11 ; in: 637906246715 ; out: 637,906,246,715.00 
ix: 12 ; in: 6379062467152 ; out: 6,379,062,467,152.00 
ix: 13 ; in: 63790624671522 ; out: 63,790,624,671,522.00 
round() overflow. Try smaller precision or use Math::BigFloat at test.pl line 11. 
at /usr/local/share/perl/5.18.2/Number/Format.pm line 535. 
    Number::Format::round('Number::Format=HASH(0x9d0b6cc)', 637906246715226, 2) called at /usr/local/share/perl/5.18.2/Number/Format.pm line 601 
    Number::Format::format_number('Number::Format=HASH(0x9d0b6cc)', 637906246715226, 2, 1) called at test.pl line 11 
Debugged program terminated. Use q to quit or R to restart, 
use o inhibit_exit to avoid stopping after program termination, 
h q, h R or h o to get additional info. 
    DB<1> p $ix   

    DB<2> 

... ale kiedy to się nie powiedzie, to nie „stop” na linii niedostatecznej jak powiedzieć gdb z może to zrobić program C - program ponownie się kończy, a więc nie mam już żadnych zmiennych kontekstowych do kontroli.

oczywiście pętla jak to może trwać tysiące razy, dlatego ustawienie przerwania na linii problematycznej i robi c ontinue ręcznie nie pomoże dużo tutaj ...

Czy istnieje sposób mieć debugera Perla, złamać program po błędzie/wyjątku, tak, że lokalny kontekst zmiennej jest zachowany, aby sprawdzić tam zmienne?

Odpowiedz

5

Wrap linia wykraczająca w eval i ustawić $DB::single gdy [email protected] jest ustawiona:

#!/usr/bin/env perl 
use warnings; 
use strict; 
use Number::Format qw(:subs); # sudo perl -MCPAN -e 'install Number::Format' 

my $tstr = ""; 
my $numFormatter = new Number::Format(); 

for (my $ix=0; $ix<20; $ix++) { 
    $tstr = $tstr . int(rand(10)); 
    my $ftstr = eval { $numFormatter->format_number($tstr, 2, 1); }; 
    $DB::single = 1 if [email protected]; 
    print "ix: $ix ; in: $tstr ; out: $ftstr\n"; 
} 

Następnie

% perl -d test.pl 

Loading DB routines from perl5db.pl version 1.49 
Editor support available. 

Enter h or 'h h' for help, or 'man perldebug' for more help. 

main::(test.pl:6): my $tstr = ""; 

    DB<1> r 
ix: 0 ; in: 7 ; out: 7.00 
ix: 1 ; in: 71 ; out: 71.00 
ix: 2 ; in: 715 ; out: 715.00 
ix: 3 ; in: 7153 ; out: 7,153.00 
ix: 4 ; in: 71537 ; out: 71,537.00 
ix: 5 ; in: 715379 ; out: 715,379.00 
ix: 6 ; in: 7153794 ; out: 7,153,794.00 
ix: 7 ; in: 71537941 ; out: 71,537,941.00 
ix: 8 ; in: 715379417 ; out: 715,379,417.00 
ix: 9 ; in: 7153794174 ; out: 7,153,794,174.00 
ix: 10 ; in: 71537941740 ; out: 71,537,941,740.00 
ix: 11 ; in: 715379417408 ; out: 715,379,417,408.00 
ix: 12 ; in: 7153794174086 ; out: 7,153,794,174,086.00 
ix: 13 ; in: 71537941740864 ; out: 71,537,941,740,864.00 
main::(test.pl:13): print "ix: $ix ; in: $tstr ; out: $tstr\n"; 

    DB<1> print $tstr 
715379417408646 

    DB<2> 

co do cholery ??

Za magii są dwie zasady:

  1. Zapobieganie wyjątki od bycia fatalne (wyjątki np chwytnych)
  2. zatrzymywania debuggera w pewnym momencie w kodzie bez powtarzanych kroków

Aby wychwycić wyjątek, użyj konstrukcji eval BLOCK. Spowoduje to zapisanie wyjątku w zmiennej [email protected]. Jeśli [email protected] nie jest pustym łańcuchem, zgłoszony został wyjątek. Zauważ, że powyższy kod, choć idiomatyczny, nie jest całkiem poprawny; jeśli zgłoszony wyjątek był ciągiem 0, byłby on ignorowany (ponieważ if 0 byłby fałszywy). Obsługa wyjątków w Perlu jest złożona. Try::Tiny ma dobrą dyskusję.

Teraz, gdy wyjątek nie jest już krytyczny, jak zatrzymać debugger? Jest na to kilka sposobów. W tym przykładzie użyto zmiennej $DB::single, która w przypadku wartości true sygnalizuje zatrzymanie debuggera. Minusem jest to, że musisz zmodyfikować swój kod, aby wywołać to zachowanie. Inną opcją jest set a breakpoint with a condition:

% perl -d test.pl 

Loading DB routines from perl5db.pl version 1.49 
Editor support available. 

Enter h or 'h h' for help, or 'man perldebug' for more help. 

main::(test.pl:6): my $tstr = ""; 

    DB<1> b 11 [email protected] ne '' 
    DB<2> r 
[... output as above ...] 
main::(test.pl:11): my $ftstr = eval { $numFormatter->format_number($tstr, 2, 1); }; 

    DB<2> p $tstr 
3247014520717436 

Zobacz perl debugger documentation więcej.

+0

To dobrze, ale to tylko dla bardzo specyficznej debugowania. Możesz rozwinąć '$ DB :: single'. – simbabque

+0

Fantastyczne - dziękuję za to, @DiabJerius – sdbbs

+0

@simbabque Rozumiem twój punkt widzenia. Ale zainteresowana strona powie "co do cholery jest to" eval "i ten szum linii' $ @ 'i' $ DB :: singiel' ?? " i sprawdzi to i dowie się o wiele więcej, niż mogę tutaj. A partia, która nie jest zainteresowana, będzie ładować ją kultowo, co stanie się niezależnie. Niektóre linki do dokumentacji są odpowiednie i dodam je. –

5

Złapanie matrycy za pomocą eval{} i $DB::single = 1 if [email protected]; to fajna sztuczka, ale jeśli chcesz dostać się do źródła problemu w bibliotece, musisz ustawić punkt przerwania w miejscu, które zgłasza błąd.

Tutaj odbywa się

DB<1> b Carp::croak 
    DB<2> c 
Carp::croak(/opt/local/lib/perl5/5.22/Carp.pm:166): 
166: sub croak { die shortmess @_ } 
    DB<3> T 
@ = DB::DB called from file '/opt/local/lib/perl5/5.22/Carp.pm' line 166 
. = Carp::croak('round() overflow. Try smaller precision or use Math::BigFloat') 
    called from file '/opt/local/lib/perl5/site_perl/5.22/Number/Format.pm' line 535 
$ = Number::Format::round(ref(Number::Format), 637906246715227, 2) 
    called from file '/opt/local/lib/perl5/site_perl/5.22/Number/Format.pm' line 601 
$ = Number::Format::format_number(ref(Number::Format), 637906246715227, 2, 1) 
    called from file 'mytest.pl' line 12 
+0

Eleganckie rozwiązanie. –