Python 异常处理的实例详解


Posted in Python onSeptember 11, 2017

Python 异常处理的实例详解

与许多面向对象语言一样,Python 具有异常处理,通过使用 try...except 块来实现。

Note: Python v s. Java 的异常处理

Python 使用 try...except 来处理异常,使用 raise 来引发异常。Java 和 C++ 使用 try...catch 来处理异常,使用 throw 来引发异常。

异常在 Python 中无处不在;实际上在标准 Python 库中的每个模块都使用了它们,并且 Python 自已会在许多不同的情况下引发它们。在整本书中你已经再三看到它们了。

• 使用不存在的字典关键字    将引发 KeyError 异常。
• 搜索列表中不存在的值    将引发 ValueError 异常。
• 调用不存在的方法    将引发 AttributeError 异常。
• 引用不存在的变量    将引发 NameError 异常。
• 未强制转换就混用数据类型    将引发 TypeError 异常。

在这些情况下,我们都在简单地使用 Python IDE:一个错误发生了,异常被打印出来 (取决于你的 IDE,可能会有意地以一种刺眼的红色形式表示),这便是。这叫做未处理异常;当异常被引发时,没有代码来明确地关注和处理它,所以异常被传给置在 Python 中的缺省的处理,它会输出一些调试信息并且终止运行。在 IDE 中,这不是什么大事,但是如果发生在你真正的 Python 程序运行的时候,整个程序将会终止。

然而,一个异常不一定会引起程序的完全崩溃。当异常引发时,可以被处理掉。有时候一个异常实际是因为代码中的 bug (比如使用一个不存在的变量),但是许多时候,一个异常是可以预见的。如果你打开一个文件,它可能不存在。如果你连接一个数据库,它可能不可连接或没有访问所需的正确的安全证书。如果知道一行代码可能会引发异常,你应该使用一个 try...except 块来处理异常。

1. 打开一个不存在的文件

>>> fsock = open("/notthere", "r")   (1) 
Traceback (innermost last): 
 File "<interactive input>", line 1, in ? 
IOError: [Errno 2] No such file or directory: '/notthere' 
>>> try: 
...   fsock = open("/notthere")    (2) 
... except IOError:           (3) 
...   print "The file does not exist, exiting gracefully" 
... print "This line will always print" (4) 
The file does not exist, exiting gracefully 
This line will always print

(1)  使用内置 open 函数,我们可以试着打开一个文件来读取 (在下一节有关于open 的更多内容)。但是那个文件不存在,所以这样就引发 IOError 异常。因为我们没有提供任何显式的对 IOError 异常的检查,Python 仅仅打印出某个关于发生了什么的调试信息,然后终止。

(2)  我们试图打开同样不存在的文件,但是这次我们在一个 try...except 内来执行它。

(3)  当 open 方法引发 IOError 异常时,我们已经准备好处理它了。except IOError: 行捕捉异常,接着执行我们自已的代码块,这个代码块在本例中只是打印出更令人愉快的错误信息。

(4)  一旦异常被处理了,处理通常在 try...except 块之后的第一行继续进行。注意这一行将总是打印出来,无论异常是否发生。如果在你的根目录下确实有一个叫 notthere 的文件,对 open 的调用将成功,except 子句将忽略,并且最后一行仍将执行。
异常可能看上去不友好 (毕竟,如果你不捕捉异常,整个程序将崩溃),但是考虑一下别的方法。你该不会希望获得一个指向不存在的文件的对象吧?不管怎么样你都得检查它的有效性,而且如果你忘记了,你的程序将会在下面某个地方给出奇怪的错误,这样你将不得不追溯到源程序。我确信你做过这种事;这可并不有趣。使用异常,一发生错误,你就可以在问题的源头通过标准的方法来处理它们。

2. 为其他用途使用异常

除了处理实际的错误条件之外,对于异常还有许多其它的用处。在标准 Python 库中一个普通的用法就是试着导入一个模块,然后检查是否它能使用。导入一个并不存在的模块将引发一个 ImportError 异常。你可以使用这种方法来定义多级别的功能??依靠在运行时哪个模块是有效的,或支持多种平台 (即平台特定代码被分离到不同的模块中)。你也能通过创建一个从内置的 Exception 类继承的类定义你自己的异常,然后使用 raise 命令引发你的异常。如果你对此感兴趣,请看进一步阅读的部分。

下面的例子演示了如何使用异常支持特定平台功能。代码来自 getpass 模块,一个从用户获得口令的封装模块。获得口令在 UNIX、Windows 和 Mac OS 平台上的实现是不同的,但是这个代码封装了所有的不同之处。

# Bind the name getpass to the appropriate function 
try: 
  import termios, TERMIOS           (1) 
except ImportError: 
  try: 
    import msvcrt              (2) 
  except ImportError: 
    try: 
      from EasyDialogs import AskPassword (3) 
    except ImportError: 
      getpass = default_getpass      (4) 
    else:                  (5) 
      getpass = AskPassword 
  else: 
    getpass = win_getpass 
else: 
  getpass = unix_getpass

(1)  termios 是 UNIX 独有的一个模块,它提供了对于输入终端的底层控制。如果这个模块无效 (因为它不在你的系统上,或你的系统不支持它),则导入失败,Python 引发我们捕捉的 ImportError 异常。

(2)  OK,我们没有 termios,所以让我们试试 msvcrt,它是 Windows 独有的一个模块,可以提供在 Microsoft Visual C++ 运行服务中的许多有用的函数的一个 API。如果导入失败,Python 会引发我们捕捉的 ImportError 异常。

(3)  如果前两个不能工作,我们试着从 EasyDialogs 导入一个函数,它是 Mac OS 独有的一个模块,提供了各种各样类型的弹出对话框。再一次,如果导入失败,Python 会引发一个我们捕捉的 ImportError 异常。

(4)  这些平台特定的模块没有一个有效 (有可能,因为 Python 已经移植到了许多不同的平台上了),所以我们需要回头使用一个缺省口令输入函数 (这个函数定义在 getpass 模块中的别的地方)。注意我们在这里所做的:我们将函数 default_getpass 赋给变量 getpass。如果你读了官方 getpass 文档,它会告诉你 getpass 模块定义了一个 getpass 函数。它是这样做的:通过绑定 getpass 到正确的函数来适应你的平台。然后当你调用 getpass 函数时,你实际上调用了平台特定的函数,是这段代码已经为你设置好的。你不需要知道或关心你的代码正运行在何种平台上;只要调用 getpass,则它总能正确处理。

(5)  一个 try...except 块可以有一条 else 子句,就像 if 语句。如果在 try 块中没有异常引发,然后 else 子句被执行。在本例中,那就意味着如果 from EasyDialogs import AskPassword 导入可工作,所以我们应该绑定 getpass 到 AskPassword 函数。其它每个 try...except 块有着相似的 else 子句,当我们发现一个 import 可用时,就绑定 getpass 到适合的函数。

如有疑问请留言或者到本站社区交流讨论,感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

Python 相关文章推荐
python基于windows平台锁定键盘输入的方法
Mar 05 Python
python文件写入实例分析
Apr 08 Python
使用Python的Zato发送AMQP消息的教程
Apr 16 Python
python实现的守护进程(Daemon)用法实例
Jun 02 Python
Python中方法链的使用方法
Feb 23 Python
Python实现通过文件路径获取文件hash值的方法
Apr 29 Python
解决pycharm 误删掉项目文件的处理方法
Oct 22 Python
Python 实现取矩阵的部分列,保存为一个新的矩阵方法
Nov 14 Python
Python判断字符串是否xx开始或结尾的示例
Aug 08 Python
Python3视频转字符动画的实例代码
Aug 29 Python
Pandas 缺失数据处理的实现
Nov 04 Python
解决Pyinstaller打包软件失败的一个坑
Mar 04 Python
Python基于回溯法子集树模板解决马踏棋盘问题示例
Sep 11 #Python
Python基于回溯法子集树模板解决找零问题示例
Sep 11 #Python
详解 Python 与文件对象共事的实例
Sep 11 #Python
Python 私有函数的实例详解
Sep 11 #Python
Python模拟用户登录验证
Sep 11 #Python
Python模拟三级菜单效果
Sep 11 #Python
轻量级的Web框架Flask 中模块化应用的实现
Sep 11 #Python
You might like
php输出xml属性的方法
2015/03/19 PHP
php中使用websocket详解
2016/09/23 PHP
php设计模式之抽象工厂模式分析【星际争霸游戏案例】
2020/01/23 PHP
javascript中String类的subString()方法和slice()方法
2011/05/24 Javascript
jQuery插件原来如此简单 jQuery插件的机制及实战
2012/02/07 Javascript
JQuery DataTable删除行后的页面更新利用Ajax解决
2013/05/17 Javascript
jQuery函数map()和each()介绍及异同点分析
2014/11/08 Javascript
JS实现在页面随时自定义背景颜色的方法
2015/02/27 Javascript
JavaScript实现的搜索及高亮显示功能示例
2017/08/14 Javascript
微信小程序分享功能之按钮button 边框隐藏和点击隐藏
2018/06/14 Javascript
对angular4子路由&amp;辅助路由详解
2018/10/09 Javascript
基于vue实现圆形菜单栏组件
2019/07/05 Javascript
jquery使用echarts实现有向图可视化功能示例
2019/11/25 jQuery
webpack的 rquire.context用法实现工程自动化的方法
2020/02/07 Javascript
vue 输入电话号码自动按3-4-4分割功能的实现代码
2020/04/30 Javascript
9种python web 程序的部署方式小结
2014/06/30 Python
基于Linux系统中python matplotlib画图的中文显示问题的解决方法
2017/06/15 Python
Pycharm 设置自定义背景颜色的图文教程
2018/05/23 Python
win10子系统python开发环境准备及kenlm和nltk的使用教程
2019/10/14 Python
如何获取Python简单for循环索引
2019/11/21 Python
Python解释器以及PyCharm的安装教程图文详解
2020/02/26 Python
python利用opencv实现SIFT特征提取与匹配
2020/03/05 Python
Python爬虫之爬取淘女郎照片示例详解
2020/07/28 Python
sublime3之内网安装python插件Anaconda的流程
2020/11/10 Python
Python代码覆盖率统计工具coverage.py用法详解
2020/11/25 Python
Forever 21美国官网:美国标志性快时尚品牌
2017/02/20 全球购物
Expedia印度尼西亚站:预订酒店、廉价航班和度假套餐
2018/01/31 全球购物
CLR与IL分别是什么含义
2016/08/23 面试题
大学生求职信范文
2014/05/24 职场文书
体育教师求职信
2014/06/30 职场文书
汽修专业自荐信
2014/07/07 职场文书
镇党政领导班子民主生活会思想汇报
2014/10/11 职场文书
工会2014法制宣传日活动总结
2014/11/01 职场文书
诚信教育主题班会
2015/08/13 职场文书
幼儿园心得体会范文
2016/01/21 职场文书
Spring Bean是如何初始化的详解
2022/03/22 Java/Android