分享

四、Python try except else详解

 星光闪亮图书馆 2019-08-08
Python 中,还有另一种异常处理结构,就是 try except else 语句,也就是在原来 try except 语句的基础上再添加一个 else 子句,其作用是指定当 try 块中没有发现异常时要执行的代码。换句话说,当 try 块中发现异常,则 else 块中的语句将不会被执行。

例如如下程序:
  1. s = input('请输入除数:')
  2. try:
  3. result = 20 / int(s)
  4. print('20除以%s的结果是: %g' % (s , result))
  5. except ValueError:
  6. print('值错误,您必须输入数值')
  7. except ArithmeticError:
  8. print('算术错误,您不能输入0')
  9. else:
  10. print('没有出现异常')
  11. print("程序继续运行")
上面程序为异常处理流程添加了 else 块,当程序中的 try 块没有出现异常时,程序就会执行 else 块。运行上面程序,如果用户输入导致程序中的 try 块出现了异常,则运行结果如下:

请输入除数:a
值错误,您必须输入数值
程序继续运行

如果用户输入让程序中的 try 块顺利完成,则运行结果如下:

请输入除数:3
20 除以3 的结果是:6.66667
没有出现异常
程序继续运行

看到这里,可能有读者觉得奇怪,既然只有当 try 块没有异常时才会执行 else 块,那么直接把 else 块的代码放在 try 块的代码的后面不就行了?

实际上大部分语言的异常处理都没有 else 块,它们确实是将 else 块的代码直接放在 try 块的代码的后面的,因为对于大部分场景而言,直接将 else 块的代码放在 try 块的代码的后面即可。

但 Python 的异常处理使用 else 块绝不是多余的语法,当 try 块没有异常,而 else 块有异常时,就能体现出 else 块的作用了。例如如下程序:
  1. def else_test():
  2. s = input('请输入除数:')
  3. result = 20 / int(s)
  4. print('20除以%s的结果是: %g' % (s , result))
  5. def right_main():
  6. try:
  7. print('try块的代码,没有异常')
  8. except:
  9. print('程序出现异常')
  10. else:
  11. # 将else_test放在else块中
  12. else_test()
  13. def wrong_main():
  14. try:
  15. print('try块的代码,没有异常')
  16. # 将else_test放在try块代码的后面
  17. else_test()
  18. except:
  19. print('程序出现异常')
  20. wrong_main()
  21. right_main()
上面程序中定义了一个 else_test() 函数,该函数在运行时需要接收用户输入的参数,随着用户输入数据的不同可能导致异常。接下来程序定义了 right_main() 和 wrong_main() 两个函数,其中 right_main() 将 else_test() 函数放在 else 块内;而 wrong_main() 将 else_test() 函数放在 try 块的代码的后面。

正如上面所介绍的,当 try 块和 else 块都没有异常时,将 else_test() 函数放在 try 块的代码的后面和放在 else 块中没有任何区别。例如,如果用户输入的数据没有导致程序出现异常,则将看到程序产生如下输出结果:

try块的代码,没有异常
请输入除数:4
20除以4的结果是: 5
try块的代码,没有异常
请输入除数:4
20除以4的结果是: 5

但如果用户输入的数据让 else_test() 函数出现异常(try 块依然没有任何异常),此时程序就会产生如下输出结果:

try块的代码,没有异常
请输入除数:0
程序出现异常
try块的代码,没有异常
请输入除数:0
Traceback (most recent call last):
  File "C:\Users\mengma\Desktop\1.py", line 21, in <module>
    right_main()
  File "C:\Users\mengma\Desktop\1.py", line 12, in right_main
    else_test()
  File "C:\Users\mengma\Desktop\1.py", line 3, in else_test
    result = 20 / int(s)
ZeroDivisionError: division by zero

对比上面两个输出结果,用户输入的都是 0,这样都会导致 else_test() 函数出现异常。如果将 else_test() 函数放在 try 块的代码的后面,此时 else_test() 函数运行产生的异常将会被 try 块对应的 except 捕获,这正是 Python 异常处理机制的执行流程:但如果将 else_test() 函数放在 else 块中,当 else_test() 函数出现异常时,程序没有 except 块来处理该异常,该异常将会传播给 Python 解释器,导致程序中止。

对比上面两个输出结果,不难发现,放在 else 块中的代码所引发的异常不会被 except 块捕获。

所以,如果希望某段代码的异常能被后面的 except 块捕获,那么就应该将这段代码放在 try 块的代码之后;如果希望某段代码的异常能向外传播(不被 except 块捕获),那么就应该将这段代码放在 else 块中。

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多