异常处理
#异常处理: try/except:捕捉由代码中的异常并恢复,匹配except里面的错误,并执行except中定义的代码,后继续执行程序(发生异常后,由except捕捉到异常后,不会中断程序,继续执行try语句后面的程序) try/finally: 无论异常是否发生,都执行清理行为 (发生异常时程序会中断程序,只不过会执行finally后的代码) raise: 手动在代码中接触发异常。 assert: 有条件地在程序代码中触发异常。 with/as 实现环境管理器。
try/except
1.1使用try和except语句来捕获异常
捕捉由代码中的异常并恢复,匹配except里面的错误,并执行except中定义的代码,后继续执行程序(发生异常后,由except捕捉到异常后,不会中断程序,继续执行try语句后面的程序)。
try:
block
except [exception,[data…]]:
block
try的完整形式:try/多个except/else语句
else是可选的
try首行底下的代码块代表此语句的主要动作:试着执行的程序代码。except分句定义try代码块内引发的异常处理器,而else分句(如果有)则是提供没有发生异常时候要执行的处理器。
1.2 使用try跟finally:
无论异常是否发生,都执行清理行为 (发生异常时程序会中断程序,只不过会执行finally后的代码)
语法如下:
try:
block
finally:
block
该语句的执行规则是:
· 执行try下的代码。
· 如果发生异常,在该异常传递到下一级try时,执行finally中的代码。
· 如果没有发生异常,则执行finally中的代码。
第二种try语法在无论有没有发生异常都要执行代码的情况下是很有用的。例如我们在python中打开一个文件进行读写操作,我在操作过程中不管是否出现异常,最终都是要把该文件关闭的。这两种形式相互冲突,使用了一种就不允许使用另一种,而功能又各异。
1.3 统一try/except/finally分句
try:
main-action:
except Exception1:
hander1
except Exception2:
hander2
...
else:
else-block
finally:
finally-block
这语句中main-action代码会先执行。如果该程序代码(main-action)引发异常,那么except代码块都会逐一测试,寻找与抛出的异常相符的语句。如果引发异常的是Exception1则会执行hander1代码块,如果引发异常的是Exception2,则会执行hander2代码块。以此类推。如果没有
引发异常,将会执行else-block代码块。
无论前面发生什么,当main-action代码块完成时。finally-block都会执行。
2.用raise语句手工引发一个异常: 自定义异常
raise [exception[,data]]
在Python中,要想引发异常,最简单的形式就是输入关键字raise,后跟要引发的异常的名称。异常名称标识出具体的类:Python异常是那些类的对象。执行raise语句时,Python会创建指定的异常类的一个对象。raise语句还可指定对异常对象进行初始化的参数。为此,请在异常类的名称后添加一个逗号以及指定的参数(或者由参数构成的一个元组)。
try:
raise MyError #自己抛出一个异常
except MyError:
print 'a error'
try :
s = raw_input( 'Enter something --> ')
if len (s) < 3 :
raise ShortInputException(len (s), 3 )
# Other work can continue as usual here
except EOFError:
print '\nWhy did you do an EOF on me?'
except ShortInputException, x:
print 'ShortInputException: The input was of length %d, \
was expecting at least %d' % (x.length,x.atleast)
else:
print 'No exception was raised.'
assert可以有条件地在程序代码中触发异常,可以认为是有条件的raise.
牢记:assert几乎都是用来收集用户定义的约束条件,而不是捕捉内在的程序设计错误。因为Python会自动收集程序的设计错误,通常咩有必要写assert去捕捉超出索引值,类型不匹配以及除数为0之类的事。
引发的异常为:AssertionError。如果没有被try捕捉到,就会终止程序。
该语句形式:
assert <test>,<data>
[python]
def f(x):
assert x>0,'x must be great zerot'
return x**2
f(-1)
3.内置Exception类
Python把内置异常组织成层次,来支持各种捕捉模式
Exception: 异常的顶层根超类
StandardError: 所有内置错误异常的超类
ArithmeticError: 所有数值错误的超类
OverflowError: 识别特定的数值错误的子类
[python]
import exceptions
help(exceptions)
4. 采用sys模块回溯最后的异常
import sys
try:
block
except:
info=sys.exc_info()
print info[0],":",info[1]
或者以如下的形式:
import sys
tp,val,td = sys.exc_info()
sys.exc_info()的返回值是一个tuple, (type, value/message, traceback)
这里的type ---- 异常的类型
value/message ---- 异常的信息或者参数
traceback ---- 包含调用栈信息的对象。
从这点上可以看出此方法涵盖了traceback.
习题:
利用异常打印出行号和函数名称 利用 raise
提示:
[python]
f = sys.exc_info()[2].tb_frame.f_back
f.f_code.co_name, f.f_lineno #函数名 行号
sayHello
[python]
import sys www.2cto.com
class PrintNameLine(Exception):
def __init__ (self,say):
Exception.__init__(self)
self.sayhello = say
try :
raise PrintNameLine('helloworld')
except PrintNameLine,x:
print '%s'%(x.sayhello )
import sys
import traceback
traceback_template = '''Traceback (most recent call last): File "%(filename)s", line %(lineno)s, in %(name)s %(type)s: %(message)s\n'''
try: 1/0 except: exc_type, exc_value, exc_traceback = sys.exc_info() traceback_details = { 'filename': exc_traceback.tb_frame.f_code.co_filename, 'lineno' : exc_traceback.tb_lineno, 'name' : exc_traceback.tb_frame.f_code.co_name, 'type' : exc_type.__name__, 'message' : exc_value.message, # or see traceback._some_str() } del(exc_type, exc_value, exc_traceback) print traceback_template % traceback_details
posted on 2014-04-10 00:23 myworldworld 阅读(85) 评论(0) 收藏 举报
浙公网安备 33010602011771号