perl函数说明(eval)

来源:互联网 发布:笔记本电脑性能优化 编辑:程序博客网 时间:2024/05/21 07:13
  • eval BLOCK
  • eval EXPR
  • eval

eval 关键字在 Perl 里起两种不同的但相关的作用。这些目的是用两种形式的语法来表现的, eval BLOCK 和 eval EXPR。第一种形式捕获那些致命的运行时例外(错误),类似于 C++ 或 Java 里的 “try 块”。第二种形式在运行时实时地编译和执行一小段代码,并且也和第一种形式一样捕获任何例外。但是第二种形式比第一种形式运行的速度慢很多,因为它每次都要分析该字串。另外,它也更通用。不管你在那里使用,eval 都是一个在 Perl 里做全部例外处理的好地方。

两种形式的 eval 所返回的值都是它计算的最后一个表达式的值,这一点和子过程一样。类似的,你可以用 return 操作符从 eval 的中间返回一个数值。提供返回值的表达式是在空,标量,或者列表环境中计算的,具体哪种环境是由 eval 本身所处的环境决定的。参阅 wantarray 获取如何判断计算环境的信息。

如果有一个可捕获的错误存在(包括任何由 die 操作符生成的),eval 返回 undef 并且把错误信息放到 $@ 里。如果没有错误,Perl 保证把 $@ 设置为空字串,所以你稍后可以很可靠地做错误检查。一个简单的布尔测试就足够了:

   eval { ... };      # 捕获运行时错误   if ($@) { ... }   # 处理错误

eval BLOCK 形式是在编译时做语法检查的,所以它的效率相当高。(熟悉慢速的 eval EXPR 形式的人们可能会被这个问题搞糊涂。)因为在 BLOCK 里的代码是和周围的代码同时编译的,所以这种形式的 eval 不能捕获语法错误。

eval EXPR 形式可以捕获语法错误是因为它在运行时分析代码。(如果分析失败,它象平常一样在 $@ 里放分析错误。)另外,它把 EXPR 的值当作一小段 Perl 程序执行。这段代码是在当前 Perl 程序的环境中执行的,这就意味着它可以从包围的范围里看到任何词汇闭合域,并且在 eval 完成之后,任何非局部变量设置仍然有效,就象子过程调用或者格式定义一样。eval 的代码是当作一个块看待的,所以任何在 eval 里定义的局部范围的变量都只能持续到 eval 结束。(参阅 my 和 local。)和任何块里的代码一样,最后的分号不是必须的。

下面是一个简单的 Perl shell。它提示用户输入任意 Perl 代码字串,编译并执行该字串,并且打印出发生的任何错误:

   print "\nEnter some Perl code: ";      while () {      eval;      print $@;      print "\nEnter some more Perl code: ";   }

下面是用 Perl 表达式做大批文件改名的一个 rename 程序:

   #! /usr/bin/perl   # rename - change filenames   $op = shift;   for (@ARGV) {      $was = $_;      eval $op;      die if $@;      # 下一行调用内建函数,而不是同名的脚本      rename($was, $_) unless $was eq $_;   }

你要这样用这个程序:

   $rename 's/\.orig$//'         *.orig   $rename 'y/A-Z/a-z/ unless /^Make/'   *   $rename '$_ .= ".bad"'         *.f

因为 eval 捕获那些致命错误,所以它可以用来判断某种特性(比如 fork 和 symlink)是否实现之类的东西。

因为 eval BLOCK 是在编译时做语法检查的,所以任何语法错误都提前报告。因此,如果你的代码不会变化,并且 eval EXPR 和 eval BLOCK 都完全符合你的要求,那么 BLOCK 形式好一些。比如:

   # 零除零不是致命错误   eval { $answer = $a / $b; };   warn $@ if $@;   # 一样的东西,但是如果多次运行就没有那么高效了   eval '$answer = $a / $b';      warn $@ if $@;   # 一个编译时语法错误(不捕获)   eval { $answer = };      # 错   # 一个运行时语法错   eval '$answer =';         # 设置 $@

这里,在 BLOCK 里的代码必须是合法的 Perl 代码,这样它才能通过编译阶段。在 EXPR 里的代码直到运行时才检查,所以要到运行时它才导致错误的发生。

eval BLOCK 的块并不是循环,所以 next,last,或 redo 这样的循环控制语句并不能用于离开或者重启该块。


0 0