Python中的异常处理

2019/07/10

错误

  • 语法上的
  • 逻辑上的
  • 遇到错误后,解释器会引发异常。如果异常对象并未被处理或捕捉,程序就会用所谓的回溯(traceback)终止执行。

异常处理语句 try…except…finally

  • except语句不是必须的,finally语句也不是必须的,但是二者必须要有一个,否则就没有try的意义了。

  • except语句可以有多个,Python会按except语句的顺序依次匹配你指定的异常,如果异常已经处理就不会再进入后面的except语句。

  • except语句可以以元组形式同时指定多个异常。

  • except语句后面如果不指定异常类型,则默认捕获所有异常,你可以通过logging或者sys模块获取当前异常。

    import logging
    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
    
  • 如果要捕获异常后要重复抛出,请使用raise,后面不要带任何参数或信息。

  • 不建议捕获并抛出同一个异常,请考虑重构你的代码。

  • 不建议在不清楚逻辑的情况下捕获所有异常,有可能你隐藏了很严重的问题。

  • 尽量使用内置的异常处理语句来替换try/except语句,比如with语句,getattr()方法。

抛出异常

​使用raise关键字可自主抛出一个异常,等同于Java和C#中的throw;

raise NameError("bad name!")

raise关键字后面可以指定你要抛出的异常实例,一般来说抛出的异常越详细越好,Python在exceptions模块内建了很多的异常类型,通过使用dir()函数来查看exceptions中的异常类型,如下:

import exceptions

print dir(exceptions)
# ['ArithmeticError', 'AssertionError'...]

异常处理时常见的问题

如何自定义异常类型

直接从Exception类继承即可。

Exception和BaseException

当要捕获一个通用异常时,应该用Exception还是BaseException?以下是它们之间的继承关系。

BaseException
 +-- SystemExit
 +-- KeyboardInterrupt
 +-- GeneratorExit
 +-- Exception
      +-- StopIteration...
      +-- StandardError...
      +-- Warning...

程序在捕获所有异常时更应该使用Exception而不是BaseException,因为被排除的三个异常属于更高级别的异常,合理的做法应该是交给Python的解释器处理。

使用内置的语法范式代替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')

异常速查表

异常处理好习惯

  • 只处理你知道的异常,尽量避免捕获所有异常然后吞掉它们。
  • 抛出的异常应该说明原因,有时候你知道异常类型也猜不出所以然。
  • 不要使用异常来控制流程,那样你的程序会无比难懂和难维护。
  • 如果有需要,切记使用finally来释放资源。
  • 如果有需要,请不要忘记在处理异常后做清理工作或者回滚操作。

注:以上内容摘自betacat 《总结:Python中的异常处理》,内容有些许删改。


一只萌萌的慧宝

(转载本站文章请注明作者和出处 花开空半夏-haoforward

Show Disqus Comments