设为首页 收藏本站
查看: 1446|回复: 0

[经验分享] Python 中的异常处理

[复制链接]

尚未签到

发表于 2018-8-15 13:51:34 | 显示全部楼层 |阅读模式
  异常处理在任何一门编程语言里都是值得关注的一个话题,良好的异常处理可以让你的程序更加健壮,清晰的错误信息更能帮助你快速修复问题。在Python中,和不部分高级语言一样,使用了try/except/finally语句块来处理异常,如果你有其他编程语言的经验,实践起来并不难。
异常处理语句 try...excpet...finally
实例代码
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.')# testsdiv(2, 0)
  
div(2, 'bad type')
  
div(1, 2)# Mutiple exception in one linetry:
  
    print(a / b)except (ZeroDivisionError, TypeError) as e:
  
    print(e)# Except block is optional when there is finallytry:
  
    open(database)finally:
  
    close(database)# catch all errors and log ittry:
  
    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
总结如下

  •   except语句不是必须的,finally语句也不是必须的,但是二者必须要有一个,否则就没有try的意义了。
  •   except语句可以有多个,Python会按except语句的顺序依次匹配你指定的异常,如果异常已经处理就不会再进入后面的except语句。
  •   except语句可以以元组形式同时指定多个异常,参见实例代码。
  •   except语句后面如果不指定异常类型,则默认捕获所有异常,你可以通过logging或者sys模块获取当前异常。
  •   如果要捕获异常后要重复抛出,请使用raise,后面不要带任何参数或信息。
  •   不建议捕获并抛出同一个异常,请考虑重构你的代码。
  •   不建议在不清楚逻辑的情况下捕获所有异常,有可能你隐藏了很严重的问题。
  •   尽量使用内置的异常处理语句来 替换try/except语句,比如with语句,getattr()方法。
抛出异常 raise
  如果你需要自主抛出异常一个异常,可以使用raise关键字,等同于C#和Java中的throw语句,其语法规则如下。
raise NameError("bad name!")  raise关键字后面需要指定你抛出的异常类型,一般来说抛出的异常越详细越好,Python在exceptions模块内建了很多的异常类型,通过使用dir()函数来查看exceptions中的异常类型,如下:
import exceptions# ['ArithmeticError', 'AssertionError'.....]print dir(exceptions)  当然你也可以查阅Python的文档库进行更详细的了解。

  •   https://docs.python.org/2.7/l...
自定义异常类型
  Python中也可以自定义自己的特殊类型的异常,只需要要从Exception类继承(直接或间接)即可:
class SomeCustomException(Exception):  
    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
  如果你有兴趣了解更多,建议阅读这篇博客。

  •   http://www.ianbicking.org/blo...
  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
  
    passexcept KeyError, e:  # should not
  
    pass
  在Python2的时代,你可以使用以上两种写法中的任意一种。在Python3中你只能使用第一种写法,第二种写法被废弃掉了。第一个种写法可读性更好,而且为了程序的兼容性和后期移植的成本,请你也抛弃第二种写法。
raise "Exception string"
  把字符串当成异常抛出看上去是一个非常简洁的办法,但其实是一个非常不好的习惯。
if is_work_done():    passelse:    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 nottry:  
    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 nameexcept AttributeError:
  
    name = 'default'
  其实你可以使用更简单的getattr()来达到你的目的。
name = getattr(test, 'name', 'default')最佳实践
  最佳实践不限于编程语言,只是一些规则和填坑后的收获。

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

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-552294-1-1.html 上篇帖子: python模块paramiko与ssh-baby神 下篇帖子: python click模块-命令行神器
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表