理解Python中的With语句


Posted in Python onMarch 18, 2016

 有一些任务,可能事先需要设置,事后做清理工作。对于这种场景,Python的with语句提供了一种非常方便的处理方式。一个很好的例子是文件处理,你需要获取一个文件句柄,从文件中读取数据,然后关闭文件句柄。 Without the with statement, one would write something along the lines of: 如果不用with语句,代码如下:

file = open("/tmp/foo.txt")
data = file.read()
file.close()

这里有两个问题。一是可能忘记关闭文件句柄;二是文件读取数据发生异常,没有进行任何处理。下面是处理异常的加强版本:

file = open("/tmp/foo.txt")
try:
  data = file.read()
finally:
  file.close()

虽然这段代码运行良好,但是太冗长了。这时候就是with一展身手的时候了。除了有更优雅的语法,with还可以很好的处理上下文环境产生的异常。下面是with版本的代码:

with open("/tmp/foo.txt") as file:
  data = file.read()

with如何工作?
 这看起来充满魔法,但不仅仅是魔法,Python对with的处理还很聪明。基本思想是with所求值的对象必须有一个__enter__()方法,一个__exit__()方法。

紧跟with后面的语句被求值后,返回对象的__enter__()方法被调用,这个方法的返回值将被赋值给as后面的变量。当with后面的代码块全部被执行完之后,将调用前面返回对象的__exit__()方法。 This can be demonstrated with the following example: 下面例子可以具体说明with如何工作:

#!/usr/bin/env python
# with_example01.py
 
 
class Sample:
  def __enter__(self):
    print "In __enter__()"
    return "Foo"
 
  def __exit__(self, type, value, trace):
    print "In __exit__()"
 
 
def get_sample():
  return Sample()
 
 
with get_sample() as sample:
  print "sample:", sample

运行代码,输出如下

bash-3.2$ ./with_example01.py
In __enter__()
sample: Foo
In __exit__()

正如你看到的, 1. __enter__()方法被执行 2. __enter__()方法返回的值 - 这个例子中是"Foo",赋值给变量'sample' 3. 执行代码块,打印变量"sample"的值为 "Foo" 4. __exit__()方法被调用 with真正强大之处是它可以处理异常。可能你已经注意到Sample类的__exit__方法有三个参数- val, type 和 trace。 这些参数在异常处理中相当有用。我们来改一下代码,看看具体如何工作的。

#!/usr/bin/env python
# with_example02.py
 
 
class Sample:
  def __enter__(self):
    return self
 
  def __exit__(self, type, value, trace):
    print "type:", type
    print "value:", value
    print "trace:", trace
 
  def do_something(self):
    bar = 1/0
    return bar + 10
 
with Sample() as sample:
  sample.do_something()

这没有任何关系,只要紧跟with后面的语句所返回的对象有__enter__()和__exit__()方法即可。此例中,Sample()的__enter__()方法返回新创建的Sample对象,并赋值给变量sample。 When executed: 代码执行后:

bash-3.2$ ./with_example02.py
type: <type 'exceptions.ZeroDivisionError'>
value: integer division or modulo by zero
trace: <traceback object at 0x1004a8128>
Traceback (most recent call last):
 File "./with_example02.py", line 19, in <module>
  sample.do_something()
 File "./with_example02.py", line 15, in do_something
  bar = 1/0
ZeroDivisionError: integer division or modulo by zero

 实际上,在with后面的代码块抛出任何异常时,__exit__()方法被执行。正如例子所示,异常抛出时,与之关联的type,value和stack trace传给__exit__()方法,因此抛出的ZeroDivisionError异常被打印出来了。

开发库时,清理资源,关闭文件等等操作,都可以放在__exit__方法当中。

因此,Python的with语句是提供一个有效的机制,让代码更简练,同时在异常产生时,清理工作更简单。

以上就是关于Python中的With语句的理解,希望对大家的学习有所帮助。

Python 相关文章推荐
python文件比较示例分享
Jan 10 Python
python实现数据库跨服务器迁移
Apr 12 Python
Python实现按中文排序的方法示例
Apr 25 Python
Python多线程原理与用法实例剖析
Jan 22 Python
python 实现一次性在文件中写入多行的方法
Jan 28 Python
Django框架验证码用法实例分析
May 10 Python
python自定义时钟类、定时任务类
Feb 22 Python
python join方法使用详解
Jul 30 Python
Python多进程multiprocessing、进程池用法实例分析
Mar 24 Python
python实现学生成绩测评系统
Jun 22 Python
PyChon中关于Jekins的详细安装(推荐)
Dec 28 Python
七个非常实用的Python工具包总结
Jun 15 Python
简述Python中的进程、线程、协程
Mar 18 #Python
Python实现计算最小编辑距离
Mar 17 #Python
Python引用模块和查找模块路径
Mar 17 #Python
Python使用tablib生成excel文件的简单实现方法
Mar 16 #Python
Python保存MongoDB上的文件到本地的方法
Mar 16 #Python
Python3中的真除和Floor除法用法分析
Mar 16 #Python
学习python类方法与对象方法
Mar 15 #Python
You might like
ThinkPHP让分页保持搜索状态的方法
2014/07/02 PHP
PHP中上传多个文件的表单设计例子
2014/11/19 PHP
PHP中检索字符串的方法分析【strstr与substr_count方法】
2017/02/17 PHP
javascript知识点收藏
2007/02/22 Javascript
js 目录列举函数
2008/11/06 Javascript
javascript div 弹出可拖动窗口
2009/02/26 Javascript
JQuery学习笔记 nt-child的使用
2011/01/17 Javascript
利用JS进行图片的切换即特效展示图片
2013/12/03 Javascript
JavaScript格式化日期时间的方法和自定义格式化函数示例
2014/04/04 Javascript
JavaScript实现同时调用多个函数的方法
2015/11/09 Javascript
JavaScript自学笔记(必看篇)
2016/06/23 Javascript
Vue.js递归组件构建树形菜单
2017/12/24 Javascript
Vue中使用clipboard实现复制功能
2018/09/05 Javascript
详解小程序如何动态绑定点击的执行方法
2019/11/26 Javascript
基于javascript实现贪吃蛇经典小游戏
2020/04/10 Javascript
用Nodejs实现在终端中炒股的实现
2020/10/18 NodeJs
[19:59]2014DOTA2国际邀请赛 IG战队纪录片
2014/08/07 DOTA
[04:13]2018国际邀请赛典藏宝瓶Ⅱ饰品一览
2018/07/21 DOTA
Django中使用locals()函数的技巧
2015/07/16 Python
Python错误: SyntaxError: Non-ASCII character解决办法
2017/06/08 Python
Python构建XML树结构的方法示例
2017/06/30 Python
Python绘制频率分布直方图的示例
2019/07/08 Python
详解Windows下PyCharm安装Numpy包及无法安装问题解决方案
2020/06/18 Python
python获取本周、上周、本月、上月及本季的时间代码实例
2020/09/08 Python
call在Python中改进数列的实例讲解
2020/12/09 Python
REISS英国官网:伦敦High Street最受欢迎品牌
2016/12/21 全球购物
Otticanet美国:最顶尖的世界名牌眼镜, 能得到打折季的价格
2019/03/10 全球购物
自我鉴定范文
2013/11/10 职场文书
幼儿园中秋节活动方案2013
2014/01/29 职场文书
大学生工作自荐书
2014/06/16 职场文书
地质工程专业毕业生求职信
2014/08/08 职场文书
技术股份合作协议书
2014/10/05 职场文书
如何用PHP实现分布算法之一致性哈希算法
2021/05/26 PHP
浅析NIO系列之TCP
2021/06/15 Java/Android
Android在Sqlite3中的应用及多线程使用数据库的建议
2022/04/24 Java/Android
CSS实现鼠标悬浮动画特效
2023/05/07 HTML / CSS