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 相关文章推荐
matplotlib 输出保存指定尺寸的图片方法
May 24 Python
python中从str中提取元素到list以及将list转换为str的方法
Jun 26 Python
Python Pandas批量读取csv文件到dataframe的方法
Oct 08 Python
python: 自动安装缺失库文件的方法
Oct 22 Python
Python pycharm 同时加载多个项目的方法
Jan 17 Python
python 实现在一张图中绘制一个小的子图方法
Jul 07 Python
Django框架视图层URL映射与反向解析实例分析
Jul 29 Python
详解Python self 参数
Aug 30 Python
Pytorch中实现只导入部分模型参数的方式
Jan 02 Python
如何使用pandas读取txt文件中指定的列(有无标题)
Mar 05 Python
Python基于traceback模块获取异常信息
Jul 23 Python
教你使用pyinstaller打包Python教程
May 27 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根据ip查询所在地区(非常有用,赶集网就用到)
2013/07/01 PHP
php中常见的sql攻击正则表达式汇总
2014/11/06 PHP
php libevent 功能与使用方法详解
2020/03/04 PHP
扩展JavaScript功能的正确方法(译文)
2012/04/12 Javascript
js对文章内容进行分页示例代码
2014/03/05 Javascript
Javascript前端UI框架Kit使用指南之Kitjs简介
2014/11/28 Javascript
JS实现带关闭功能的阿里妈妈网站顶部滑出banner工具条代码
2015/09/17 Javascript
Javascript中浏览器窗口的基本操作总结
2016/08/18 Javascript
JavaScript实现瀑布流图片效果
2017/06/30 Javascript
关于TypeScript模块导入的那些事
2018/06/12 Javascript
vue中当图片地址无效的时候,显示默认图片的方法
2018/09/18 Javascript
webpack自动打包和热更新的实现方法
2019/06/24 Javascript
vue单页应用的内存泄露定位和修复问题小结
2019/08/02 Javascript
关于layui flow loading占位图的实现方法
2019/09/21 Javascript
js 计数排序的实现示例(升级版)
2020/01/12 Javascript
微信小程序点击item使之滚动到屏幕中间位置
2020/03/25 Javascript
javascript设计模式 ? 原型模式原理与应用实例分析
2020/04/10 Javascript
React生命周期原理与用法踩坑笔记
2020/04/28 Javascript
Vue解决echart在element的tab切换时显示不正确问题
2020/08/03 Javascript
js中实现继承的五种方法
2021/01/25 Javascript
基于python中theano库的线性回归
2018/08/31 Python
Python编程flask使用页面模版的方法
2018/12/28 Python
安装好Pycharm后如何配置Python解释器简易教程
2019/06/28 Python
使用 Python 处理3万多条数据只要几秒钟
2020/01/19 Python
Python切割图片成九宫格的示例代码
2020/03/10 Python
浅谈HTML5 defer和async的区别
2016/06/07 HTML / CSS
HTML5本地存储localStorage、sessionStorage基本用法、遍历操作、异常处理等
2014/05/08 HTML / CSS
西铁城美国官方网站:Citizen Watch美国
2019/11/08 全球购物
大学生开西餐厅创业计划书
2014/02/01 职场文书
品牌转让协议书
2014/08/20 职场文书
国际贸易实训报告
2014/11/05 职场文书
2014年平安建设工作总结
2014/11/19 职场文书
《称赞》教学反思
2016/02/17 职场文书
导游词之阆中古城
2019/12/23 职场文书
Android存储中最基本的文件存储方式
2022/04/30 Java/Android
Go语言编译原理之源码调试
2022/08/05 Golang