理解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 相关文章推荐
10种检测Python程序运行时间、CPU和内存占用的方法
Apr 01 Python
Python语言实现机器学习的K-近邻算法
Jun 11 Python
修复 Django migration 时遇到的问题解决
Jun 14 Python
win10系统下Anaconda3安装配置方法图文教程
Sep 19 Python
Django Rest framework权限的详细用法
Jul 25 Python
详解如何用TensorFlow训练和识别/分类自定义图片
Aug 05 Python
关于Python内存分配时的小秘密分享
Sep 05 Python
python 读取数据库并绘图的实例
Dec 03 Python
使用python实现哈希表、字典、集合操作
Dec 22 Python
python如何支持并发方法详解
Jul 25 Python
Ubuntu权限不足无法创建文件夹解决方案
Nov 14 Python
Python开发五子棋小游戏
Apr 28 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
php sprintf()函数让你的sql操作更安全
2008/07/23 PHP
php实现单链表的实例代码
2013/03/22 PHP
基于PHP代码实现中奖概率算法可用于刮刮卡、大转盘等抽奖算法
2015/12/20 PHP
php解析mht文件转换成html的实例
2017/03/13 PHP
javascript对象的property和prototype是这样一种关系
2007/03/24 Javascript
浅析JQuery获取和设置Select选项的常用方法总结
2013/07/04 Javascript
JavaScript学习心得之概述
2015/01/20 Javascript
javascript实现淡蓝色的鼠标拖动选择框实例
2015/05/09 Javascript
jQuery validate插件submitHandler提交导致死循环解决方法
2016/01/21 Javascript
jQuery+ajax实现滚动到页面底部自动加载图文列表效果(类似图片懒加载)
2016/06/07 Javascript
JavaScript动态添加事件之事件委托
2016/07/12 Javascript
jQuery联动日历的实例解析
2016/12/02 Javascript
bootstrap modal弹出框的垂直居中
2016/12/14 Javascript
javascript监听页面刷新和页面关闭事件方法详解
2017/01/09 Javascript
NodeJS基础API搭建服务器详细过程记录
2017/04/01 NodeJs
vue组件与复用详解
2018/04/08 Javascript
mpvue 如何使用腾讯视频插件的方法
2018/07/16 Javascript
[02:17]2016完美“圣”典风云人物:Sccc专访
2016/12/03 DOTA
[00:38]TI珍贵瞬间系列(二):笑
2020/08/26 DOTA
python开发之IDEL(Python GUI)的使用方法图文详解
2015/11/12 Python
Python实现Linux命令xxd -i功能
2016/03/06 Python
python查询mysql,返回json的实例
2018/03/26 Python
Python实现的微信好友数据分析功能示例
2018/06/21 Python
python实现ID3决策树算法
2018/08/29 Python
实例详解Matlab 与 Python 的区别
2019/04/26 Python
20行python代码实现人脸识别
2019/05/05 Python
python3爬取torrent种子链接实例
2020/01/16 Python
tensorflow实现残差网络方式(mnist数据集)
2020/05/26 Python
网上常见的一份Linux面试题(多项选择部分)
2014/09/09 面试题
社会实践心得体会
2014/01/03 职场文书
租房协议书范例
2014/10/14 职场文书
2014年依法行政工作总结
2014/11/19 职场文书
优秀大学生申请书
2019/06/24 职场文书
详解Oracle块修改跟踪功能
2021/11/07 Oracle
Java的Object类的九种方法
2022/04/13 Java/Android
Python 中面向接口编程
2022/05/20 Python