理解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中 sys.argv[]的用法简明解释
Dec 20 Python
python 实现数组list 添加、修改、删除的方法
Apr 04 Python
python版opencv摄像头人脸实时检测方法
Aug 03 Python
分析经典Python开发工程师面试题
Apr 08 Python
python实现微信每日一句自动发送给喜欢的人
Apr 29 Python
python 返回一个列表中第二大的数方法
Jul 09 Python
django 2.2和mysql使用的常见问题
Jul 18 Python
Python中 CSV格式清洗与转换的实例代码
Aug 29 Python
浅谈pytorch池化maxpool2D注意事项
Feb 18 Python
Python爬虫教程之利用正则表达式匹配网页内容
Dec 08 Python
python爬虫利器之requests库的用法(超全面的爬取网页案例)
Dec 17 Python
django 认证类配置实现
Nov 11 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将时间差转换为字符串提示
2011/09/07 PHP
深入php处理整数函数的详解
2013/06/09 PHP
php实现监听事件
2013/11/06 PHP
php实现使用正则将文本中的网址转换成链接标签
2014/12/03 PHP
学习php设计模式 php实现原型模式(prototype)
2015/12/07 PHP
php-msf源码详解
2017/12/25 PHP
prototype 中文参数乱码解决方案
2009/11/09 Javascript
js创建元素(节点)示例
2014/01/02 Javascript
使用jQuery jqPlot插件绘制柱状图
2014/12/18 Javascript
js select实现省市区联动选择
2020/04/17 Javascript
angularJs中$scope数据序列化的实例
2018/09/30 Javascript
vue实现表格过滤功能
2019/09/27 Javascript
[原创]pip和pygal的安装实例教程
2017/12/07 Python
解决python写入mysql中datetime类型遇到的问题
2018/06/21 Python
Python中is和==的区别详解
2018/11/15 Python
Python OpenCV利用笔记本摄像头实现人脸检测
2020/08/20 Python
Django的性能优化实现解析
2019/07/30 Python
django项目中使用手机号登录的实例代码
2019/08/15 Python
python中用logging实现日志滚动和过期日志删除功能
2019/08/20 Python
Django实现文件上传下载功能
2019/10/06 Python
pygame实现非图片按钮效果
2019/10/29 Python
python使用opencv在Windows下调用摄像头实现解析
2019/11/26 Python
Pytorch 搭建分类回归神经网络并用GPU进行加速的例子
2020/01/09 Python
NumPy统计函数的实现方法
2020/01/21 Python
pytorch dataloader 取batch_size时候出现bug的解决方式
2020/02/20 Python
keras 简单 lstm实例(基于one-hot编码)
2020/07/02 Python
html5 input元素新特性_动力节点Java学院整理
2017/07/06 HTML / CSS
Book Depository亚太地区:一家领先的国际图书零售商
2019/05/05 全球购物
娱乐地球:Entertainment Earth
2020/01/08 全球购物
NOTINO英国:在线购买美容和香水
2020/02/25 全球购物
Auguste The Label官网:澳大利亚一家精品女装时尚品牌
2020/06/14 全球购物
电钳专业个人求职信
2014/01/04 职场文书
给学校的建议书范文
2014/05/15 职场文书
python之django路由和视图案例教程
2021/07/26 Python
Redis keys命令的具体使用
2022/06/05 Redis
SQL bool盲注和时间盲注详解
2022/07/23 SQL Server