Perl 作為一種腳本語言可以實時地生成和執(zhí)行代碼。這種特性可以把代碼的編譯推遲到運行時,所以又稱為“動態(tài)代碼”。另外, Perl 也如 Java 、 C++ 一樣提供了異常處理機制。本文將初步探討Perl 中實現(xiàn)動態(tài)代碼和異常處理機制的函數(shù): eval 。如有錯誤不足,歡迎討論和批評指正。
eval 函數(shù)可以看作是 Perl 虛擬機,它的參數(shù)就是一段 Perl 代碼。利用 'perldoc –f eval'可以獲取 eval 函數(shù)使用幫助,其中介紹了它的兩種使用方式:
復(fù)制代碼 代碼如下:
eval EXPR
EXPR 是一個的表達式,例如:
復(fù)制代碼 代碼如下:
eval "print $a" ;
eval 'print $a' . ', $b' ;
eval 1 + 3 ;
eval 'print ' . '$a + $b, "\n"' ;
eval $command;#$command = ‘print “hello Perl”'
eval $ARGV[0];
在執(zhí)行時, Perl 解釋器會首先解析表達式的值,然后將表達式值作為一條 Perl 語句插入當前執(zhí)行上下文。所以,新生成的語句與 eval 語句本身具有相同的上下文環(huán)境。這種方式中,每次執(zhí)行eval 語句,表達式都會被解析。所以,如果 eval EXPR 如果出現(xiàn)在循環(huán)中,表達式可能會被解析多次。 eval 的這種方式使得 Perl 腳本程序能實時生成和執(zhí)行代碼,從而實現(xiàn)了“動態(tài)代碼”。
復(fù)制代碼 代碼如下:
eval BLOCK
BLOCK 是一個代碼塊,例如:
復(fù)制代碼 代碼如下:
eval {print $a};
eval {$a = 1, $b = 2, $c = $a + $b};
與第一種方式不同, BLOCK 只會被解析一次,然后整個插入當前 eval 函數(shù)所在的執(zhí)行上下文。由于解析上的性能的優(yōu)勢,以及可以在編譯時進行代碼語法檢查,這種方式通常被作為 Perl 用來為一段代碼提供異常捕捉機制,雖然前一種方式也可以。
按幫助的名稱,稱 eval 的參數(shù)程序為“小程序” (mini-program) 。在兩種方式中, eval 函數(shù)的返回值都是小程序的最后一條語句的值,如果遇到 return 語句,與子例程相同。
Script1:
復(fù)制代碼 代碼如下:
#!/usr/bin/perl -w
push ( @program,'$i = 1;');
push ( @program,'$i = 3; $j = 2; $k = $i + $j');
push ( @program, '$i = 3; return 24; $k = $i + $j');
foreach $exp (@program)
{
$rtn =eval($exp);
print $rtn,"\n";
}
Output:
復(fù)制代碼 代碼如下:
1
5
24
如果小程序中有語法錯誤、運行時錯誤遇到 die 語句, eval 將返回 undef 。錯誤碼被保存在$@ 中。
Script2:
復(fù)制代碼 代碼如下:
#!/usr/bin/perl -w
push ( @program, '$i = 3; die "error message"; $k = $i + $j');
foreach $exp (@program)
{
$rtn =eval($exp);
if ( ! defined ( $rtn))
{
print "Exception: " , $@,"\n";
}
else
{
print $rtn,"\n";
}
} ;
Output:
復(fù)制代碼 代碼如下:
Exception: error message at (eval 1) line 1.
Script3:
復(fù)制代碼 代碼如下:
#!/usr/bin/perl -w
# a run-time error
eval '$answer =' ; # sets $@
warn $@ if$@;
Output:
復(fù)制代碼 代碼如下:
syntax error at (eval 1) line 2, at EOF