【转载】Perl异常处理方法总结
程序脚本在运行过程中,总会碰到这样那样的问题,我们会预知一些问题并为其准备好处理代码,而有一些不能预知。好的程序要能尽可能多的处理可能出现的异常问题,本文就总结了一些方法来解决这些异常,当然perl在这个处理了不及其它同类语言,但也不会差到那里。在开始前,我们先盘点一些关于perl的优缺点。
0. 历史太悠久了。你可以在1997年的计算机上找到perl5.0。(只是吐槽一下,历史悠久没什么不好,与时俱进才是关键)
1. 不回收循环垃圾(这是个硬伤,也许和Perl设计的初衷有关,小脚本影响不大;但因为这个,perl与稍微大一点的程序就无缘了)
2. 容错(如字符串和数值的隐式转换等),但换句话说就是对错误过于放纵,程序产生错误结果而不是报错。
3. 一件事有很多方法做(TIMTOWTDY),但换句话说就是做一件事没有“一种明显的做法”。比如数组的长度是$#somearray+1。为什么是加一呢?难道不是somearray.length或者len(somearray)吗?
4. 语法比较难懂。语法还处在“脚本语言”(指的是那些随手写随手扔的批处理脚本)的定位上,只是稍微比shell脚本更像传统的编程语言。比如:
4.1 大量使用$符号,以及各种用符号表示的特殊变量。
4.2 函数不带形参列表。
4.3 面向对象支持不好:语法中没有对象语法(没有class等关键字),全靠hash和bless。继承靠ISA,也没有语法支持。
4.4 只有基本的异常处理(perl的eval和die,类似java的try和throw)。但是异常只是一条字符串信息,少了基于对象的异常系统,异常处理总是不如其它语言。
以上这些都会影响可读性和可写性。需要注意的是,人的大脑同时可以关注的信息是有限的。如果写业务逻辑,却偏偏要不断关注实现细节(比如使用package, hash, ISA, bless来实现面向对象),程序员的思路就会不断地被这些细节打乱,急剧降低编程速度,还会不断犯错误,甚至为了简单性而牺牲效率甚至正确性。
一个高级的语言就是对高级的概念的抽象,使得程序员可以将思维从下层的细节中脱离出来(Separation of Concern)。Perl作为文本批处理用的脚本语言是够的,可以做很多shell不擅长的字符串处理(正则表达式很赞)和计算,这在当时也是好的。但是对于稍微大一些的程序,甚至需要面向对象的时候,Perl就不合适了。
Perl可以写出很简短的代码(谷歌一下“code golf”),在这一点上Ruby更像Perl。下面就介绍一些让Perl脚本编写的更加规范,甚至在出现错误时能得到很好的处理的方法技巧。
打开约束指令,让编码更规范
如果你使用perl5.10 或更高的版本,可以显示的指定当前Perl版本号来自动打开约束指令。
use 5.012; #自动启用use strict 指令
use v5.10; #自动布严格模式
之前的版本通过添加下边的指令:
use strict;
通过启用约束指令,编程时常见的错误很容易暴露出来。而启用warnings指令的话,还能捕获一些其他不是很紧要的问题。
如果担心原先的程序启用strict后程序不正常运行,则可以在实际修改代码前,先在命令行上试着启用strict看看:
perl -Mstrict freeoa_program.pl
Perl的约束集包括vars(变量),subs(子程序)和refs(引用)这三部分,通常这三种约束同时使用。
在很多情况下,系统调用可能会失败;例如,尝试打开不存在的文件,或者删除某个仍含有文件的目录,或者尝试读取没有读权限的文件。在前面的示例中,我们已经用到了die函数,详细讨论有关错误处理和错误处理函数的相关内容。这些函数包括die函数、warn函数和eval函数。
die函数用于在命令或文件句柄失败时退出Perl脚本。
warn函数类似于die函数,但它不会退出脚本。
eval函数具有多种用途,但它主要还是用于异常处理。
读者想必还记得短路运算符&&和||,这两个运算符首先会求其左侧操作数的值,然后才会求其右侧操作数的值。如果&&左侧操作数值为true,则求其右侧的操作数。如果||左侧操作数的值为false,这才求其右侧的操作数。
Perl 5提供的Carp模块扩展了die和warn的功能
对一段代码中如果有一句出现异常,但事先并不知道是哪一句,怎样进行异常的捕获?
eval{.........};#捕获运行时的错误
if($@){........};#进行错误处理
通过使用 Perl 的 eval 语句来分析错误,使用标准方法来处理 Perl 错误,请使用以下语法:
eval {enter statements you want to monitor};
在运行时,如果 Perl 引擎在 eval 块内的语句中遇到错误,那么它会跳过 eval 块的剩余部分,并为对应的错误文本设置 $@。例如:
eval{$objectName->MethodName();};
if ($@){
print "Error using MethodName method. Error: $@\n";
}
else{
# continue without error ...
}
某些预期通常失败的函数不包含在此范围内。尤其是验证和设置字段函数返回错误说明,而不是抛出异常。这样可拦截内部中断信号后对其进行处理,而不是简单地让程序对异常按默认方法进行处理。
The recommended way to do some things, like timeouts, is through an eval/die pair. However, I've noticed that if you set $SIG{__DIE__} to do some custom reporting, like in:
$SIG{__WARN__} = sub { ... }; # custom error report
$SIG{__DIE__} = sub { &{$SIG{__WARN__}; exit(1); };
# custom error report and terminate
eval {
$SIG{ALRM} = sub { die "timeout\n"; };
alarm 20;
...
alarm 0;
}
"die" isn't caught! It remains fatal!
The only way around this (that I found), is to clear $SIG{__DIE__} in the eval blok:
eval{
local($SIG{__DIE__}); #back to standard Perl die
...
}
Question: is there another way to terminate a __DIE__ sub, that is NOT fatal in eval? "die" doesn't die in eval, but ONLY if you didn't set $SIG{__DIE__} yourself (even outside of the eval block). Surely, this can't be the way it's supposed to be?
你最好要使用Try::Tiny模块来处理这些问题,这将帮助你避免一些老的版本里的陷阱。
use Try::Tiny;
try{
die "foo";
}catch{
warn "caught error: $_";
};
有时我们常常写多个 open ,然后还要写上多次 die 像下面,读一个文件,然后写一个文件,有时我写的 open 会超过 4 个,要写很多次 die 这时非常麻烦。我们如果直接 use autodie 就直接能解决所有问题,所有的 die 的地方都能自动实现。
open my $fh, '<', $in or die "$!";
open my $fh, '>', $out or die "$!";
用autodie简化错误处理
autodie编译指令(从5.10.1起开始自带,也可以直接从CPAN安装)
默认情况下,autodie会对它能起作用的所有函数生效。如果只是希望对某些特定函数起作用,可以将各个函数的名字或一组函数的组名列出来告诉autodie:
use autodie qw(open close); #只对特定函数生效
use autodie qw(:filesys); #只对 某组函数生效
在autodie捕获错误时,它会把$@设置为autodie::exception对象,而$@就是表示eval错误变量
加上 autodie 以后就成了
use autodie
open my $fh, '<', $fin;
open my $fh, '>', $fout;
有了这个方便多了,有时我们常常不想让程序 die 了后退出,想抓到原因,得使用 eval 然后来检查 $@,像下面这样。
use autodie;
eval{
open my $fh, '<', $in;
# .....
}
if($@){
#TODO
}
eval{
open my $fh, '>', $out;
}
if($@){
#TODO
}
还好上次只有二个 open 。是不是感觉很痛苦,要是 open 之类会 die 的更加多点的话。另外,有没有发现,只要第一个 open 失败,第二个 if ($@) 永远会失败,因为 $@ 的标记下面也检查同样的变量。这时使用 Try::Tiny 中的 try catch 吧。
与Try::Tiny模块配合使用
Perl 没有内置的异常处理机制,所以最合适的方法就是使用Try::Tiny模块。虽然CPAN中处理异常的模块很多,但是这个模块最为轻巧,使用起来也没有过多的依赖关系。
use autodie;
use Try::Tiny;
# handle errors with a catch handler
try{
die "foo";
}catch{
warn "caught error: $_"; # not $@
};
注意:catch 代码以分号结尾的,是一个表达式。另外它会将出错的信息保存在变量$_而不是$@中。
这时使用很方便
use autodie;
use Try::Tiny;
try {
open my $fh, '<', $in;
# .....
open my $fh, '>', $out;
# .....
}
catch {
print "Error $_\n";
}
有时我们还可以在这个地方使用'try catch'做程序流程控制
use autodie;
use Try::Tiny;
try {
&sub1();
&sub2();
&sub3();
}
catch {
&suba();
&subb();
&subc();
}
像上面,只要在 sub1, sub2, sub3 任意一个子函数 die 出来,这个流程就认为出错,就做其它的另一种方案的流程 suba,subb,subc。
【转载】Perl异常处理方法总结的更多相关文章
- python pip 出现locations that require TLS/SSL异常处理方法
python pip 出现locations that require TLS/SSL异常处理方法 转载 郑才华 发布于2018-03-24 21:41:16 阅读数 51844 收藏 展开 最近在r ...
- Mac终端使用swift REPL异常处理方法
Mac终端使用swift REPL异常处理方法 终端使用swift命令出现 warning: Swift error in module libmarisa.dylibDebug info from ...
- PL/SQL异常处理方法
PL/SQL异常处理方法 1:什么是异常处理: PL/SQL提供一个功能去处理异常,在PL/SQL块中叫做异常处理,使用异常处理我们能够测试代码和避免异常退出. PL/SQL异常信息包含三个部分: ...
- [转贴]从零开始学C++之异常(一):C语言错误处理方法、C++异常处理方法(throw, try, catch)简介
一.C语言错误处理方法 1.返回值(if … else语句判断错误) 2.errno(linux 系统调用) 3.goto语句(函数内局部跳转) 4.setjmp.longjmp(Do not use ...
- [置顶] Ajax程序:处理异步调用中的异常(使用Asp.Net Ajax内建的异常处理方法)
无论在Window应用程序,还是Web应用程序以对用户友好的方式显示运行时的异常都是很有必要,尤其对于可能有很多不确定因素导致异常的Web应用程序;在传统的Web开发中,处理异常的方式——设计专门一个 ...
- 从零开始学C++之异常(一):C语言错误处理方法、C++异常处理方法(throw, try, catch)简介
一.C语言错误处理方法 1.返回值(if … else语句判断错误) 2.errno(linux 系统调用) 3.goto语句(函数内局部跳转) 4.setjmp.longjmp(Do not use ...
- C语言错误处理方法、C++异常处理方法(throw, try, catch)简介
一.C语言错误处理方法 1.返回值(if … else语句判断错误) 2.errno(linux 系统调用) 3.goto语句(函数内局部跳转) 4.setjmp.longjmp(Do not use ...
- 2017.10.28 针对Java Web应用中错误异常处理方法的运用
针对Java Web应用中错误异常处理方法的运用 在javaweb中其异常都需要对Checked Exception之下的Exception进行继承,并且有选择地对发生的错误和异常进行处理.Java同 ...
- [转载]机器学习优化方法总结:SGD,Momentum,AdaGrad,RMSProp,Adam
[转载]机器学习优化方法总结:SGD,Momentum,AdaGrad,RMSProp,Adam https://blog.csdn.net/u010089444/article/details/76 ...
随机推荐
- Android 在Windows上安装FFmpeg程序
FFmpeg是一套可以用来记录.转换数字音频.视频,并能将其转化为流的开源计算机程序.它提供了录制.转换以及流化音视频的完整解决方案.它包含了非常先进的音频/视频编解码库libavcodec. 该程序 ...
- php 配置本地自定义域名
一.设置host文件 二.设置httpd.conf # Virtual hosts Include conf/extra/httpd-vhosts.conf 三.设置httpd-vhosts.conf ...
- jQuery基础修炼圣典—DOM篇(一)
一.DOM节点的创建 1.创建节点及节点属性 通过JavaScript可以很方便的获取DOM节点,从而进行一系列的DOM操作.但实际上一般开发者都习惯性的先定义好HTML结构,但这样就非常不灵活了. ...
- 启用Service Broker
2015-10-20 17:31 整理,未发布数据库邮件配置向导,在选择配置任务页面点击下一步时,弹出"数据库邮件依赖于 Service Broker...".点击是,整个SSMS ...
- Java File类基本操作
我们可以利用Java.io.File类对文件进行操作,基本操作如下: 1)创建文件: public boolean createNewFile() throws IOException 2)删除文件: ...
- Maven开发环境的搭建,含jetty调试[简单明了]
之前的一片老文,搬过来. 1. Maven的开发环境需要哪些部件a) Maven.去http://maven.apache.org/ 下载.将其bin目录加入进系统PATH.测试方法:用命令mvn – ...
- 学习OpenCV——鼠标事件(画框)
#include "cv.h" #include "highgui.h" bool check_line_state=false; IplImage* work ...
- 学习OpenCV——粒子滤波(网上两篇文章总结)
粒子滤波的理论实在是太美妙了,用一组不同权重的随机状态来逼近复杂的概率密度函数.其再非线性.非高斯系统中具有优良的特性.opencv给出了一个实现,但是没有给出范例,学习过程中发现网络上也找不到.le ...
- editplus3运行Python程序
editplus3是一款不错的编辑器,他可以编译,运行java,php等各种程序,现把他运行Python程序的方法贴出来,首先得安装python,然后打开editplug3,工具——配置用户工具——组 ...
- compilation filed Unable to write to path xxxxxx 遇到这种情况的话
如果是xib文件的话 查看target —>Build Phases —> Copy Bundle Resources里面查看路径. 可能有重复路径导致编译不通过.