在编程语言的世界里,几乎每一门语言都有自己的异常处理机制,python也不例外,本文主要转载自 Python中的异常处理并有部分补充。
进入正题,首先异常是什么?
异常即一个事件,该事件会在程序执行的过程中发生,影响程序的正常执行。
如何捕获?
python中是以try/except语句:用来检测try语句块中的错误,从而让exception语句捕获异常信息并处理。
接着看下官方文档给出说明示例:
#-----------------------------【1】----------------------------------
def div(a, b):
try:
print(a / b)
except ZeroDivisionError:
print("Error: b should not be 0 !!")
except Exception as e:
print("Unexpected Error: {}".format(e))
else:
print('Run into else only when everything goes well')
finally:
print('Always run into finally block.')
#-----------------------------【1】----------------------------------
# tests
div(2, 0)
div(2, 'bad type')
div(1, 2)
#-----------------------------【2】----------------------------------
# Mutiple exception in one line
try:
print(a / b)
except (ZeroDivisionError, TypeError) as e: #【3】
print(e)
# Except block is optional when there is finally
try:
open(database)
finally:
close(database)
#-----------------------------【2】----------------------------------
# catch all errors and log it
try:
do_work()
except:
# get detail from logging module
logging.exception('Exception caught!')
# get detail from sys.exc_info() method
error_type, error_value, trace_back = sys.exc_info()
print(error_value)
raise 【5】
总结如下
【1】except和finally 语句至少有一个,否则try就没了意义。
【2】except语句可以有多个,Python会按照顺序依次匹配你指定的异常,如果异常已经处理就不再进入都后面的except语句
【3】except 语句可以以元组的形式同时指定多个异常
【4】except语句后如果不指定异常类型,则默认捕获全部异常
【5】如果要捕获异常后需重复抛出,则使用raise,后不用带任何参数或信息
【6】不建议捕获并抛出同一个异常,请考虑重构你的代码。
【7】不建议在不清楚逻辑的情况下捕获所有异常,有可能你隐藏了很严重的问题。
【8】尽量使用内置的异常处理语句来替换try/except语句,比如with语句,getattr()方法。
抛出异常raise
如果你需要自主抛出一个异常,可以使用raise关键字,等同于Java中的throw,语法规则如下
raise NameError('bad name!')
raise
关键字后面可以指定你要抛出的异常实例,一般来说抛出的异常越详细越好,Python在exceptions
模块内建了很多的异常类型,通过使用dir()
函数来查看exceptions
中的异常类型,如下:
import exceptions
print dir(exceptions)
# ['ArithmeticError', 'AssertionError'...]
当然你也可以查阅Python的文档库进行更详细的了解。
自定义异常类型
Python中自定义自己的异常类型非常简单,只需要要从Exception
类继承即可(直接或间接):
class SomeCustomException(Exception):
pass
class AnotherException(SomeCustomException):
pass
一般你在自定义异常类型时,需要考虑的问题应该是这个异常所应用的场景。如果内置异常已经包括了你需要的异常,建议考虑使用内置的异常类型。比如你希望在函数参数错误时抛出一个异常,你可能并不需要定义一个InvalidArgumentError
,使用内置的ValueError
即可。
经验案例
传递异常 re-raise Exception
捕捉到了异常,但是又想重新抛出它(传递异常),使用不带参数的raise
语句即可:
def f1():
print(1/0)
def f2():
try:
f1()
except Exception as e:
raise # don't raise e !!!
f2()
在Python2中,为了保持异常的完整信息,那么你捕获后再次抛出时千万不能在raise
后面加上异常对象,否则你的trace
信息就会从此处截断。以上是最简单的重新抛出异常的做法,也是推荐的做法。
还有一些技巧可以考虑,比如抛出异常前你希望对异常的信息进行更新。
def f2():
try:
f1()
except Exception as e:
e.args += ('more info',)
raise
如果你有兴趣了解更多,建议阅读这篇博客。
Python3对重复传递异常有所改进,你可以自己尝试一下,不过建议还是遵循以上规则。
Exception 和 BaseException
当我们要捕获一个通用异常时,应该用Exception
还是BaseException
?我建议你还是看一下 官方文档,这两个异常到底有啥区别呢? 请看它们之间的继承关系。
BaseException
+-- SystemExit
+-- KeyboardInterrupt
+-- GeneratorExit
+-- Exception
+-- StopIteration...
+-- StandardError...
+-- Warning...
从Exception
的层级结构来看,BaseException
是最基础的异常类,Exception
继承了它。BaseException
除了包含所有的Exception
外还包含了SystemExit
,KeyboardInterrupt
和GeneratorExit
三个异常。
由此看来你的程序在捕获所有异常时更应该使用Exception
而不是BaseException
,因为被排除的三个异常属于更高级别的异常,合理的做法应该是交给Python的解释器处理。
except Exception as e和 except Exception, e
代码示例如下:
try:
do_something()
except NameError as e: # should
pass
except KeyError, e: # should not
pass
在Python2的时代,你可以使用以上两种写法中的任意一种。在Python3中你只能使用第一种写法,第二种写法已经不再支持。第一个种写法可读性更好,而且为了程序的兼容性和后期移植的成本,请你果断抛弃第二种写法。
raise "Exception string"
把字符串当成异常抛出看上去是一个非常简洁的办法,但其实是一个非常不好的习惯。
if is_work_done():
pass
else:
raise "Work is not done!" # not cool
上面的语句如果抛出异常,那么会是这样的:
Traceback (most recent call last):
File "/demo/exception_hanlding.py", line 48, in <module>
raise "Work is not done!"
TypeError: exceptions must be old-style classes or derived from BaseException, not str
这在 Python2.4 以前是可以接受的做法,但是没有指定异常类型有可能会让下游没办法正确捕获并处理这个异常,从而导致你的程序难以维护。简单说,这种写法是是封建时代的陋习,应该扔了。
使用内置的语法范式代替try/except
Python 本身提供了很多的语法范式简化了异常的处理,比如for
语句就处理了的StopIteration
异常,让你很流畅地写出一个循环。
with
语句在打开文件后会自动调用finally
并关闭文件。我们在写 Python 代码时应该尽量避免在遇到这种情况时还使用try/except/finally的思维来处理。
# should not
try:
f = open(a_file)
do_something(f)
finally:
f.close()
# should
with open(a_file) as f:
do_something(f)
再比如,当我们需要访问一个不确定的属性时,有可能你会写出这样的代码:
try:
test = Test()
name = test.name # not sure if we can get its name
except AttributeError:
name = 'default'
其实你可以使用更简单的getattr()
来达到你的目的。
name = getattr(test, 'name', 'default')
最佳实践
最佳实践不限于编程语言,只是一些规则和填坑后的收获。
- 只处理你知道的异常,避免捕获所有异常然后吞掉它们。
- 抛出的异常应该说明原因,有时候你知道异常类型也猜不出所以然。
- 避免在
catch
语句块中干一些没意义的事情,捕获异常也是需要成本的。 - 不要使用异常来控制流程,那样你的程序会无比难懂和难维护。
- 如果有需要,切记使用
finally
来释放资源。 - 如果有需要,请不要忘记在处理异常后做清理工作或者回滚操作。
网友评论