理解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 相关文章推荐
pyqt4教程之实现半透明的天气预报界面示例
Mar 02 Python
在centos7中分布式部署pyspider
May 03 Python
python logging日志模块以及多进程日志详解
Apr 18 Python
修复 Django migration 时遇到的问题解决
Jun 14 Python
Python实现将通信达.day文件读取为DataFrame
Dec 22 Python
pyQt5实时刷新界面的示例
Jun 25 Python
原来我一直安装 Python 库的姿势都不对呀
Nov 11 Python
Python单元测试模块doctest的具体使用
Feb 10 Python
Django Channel实时推送与聊天的示例代码
Apr 30 Python
python3+selenium获取页面加载的所有静态资源文件链接操作
May 04 Python
Python 执行矩阵与线性代数运算
Aug 01 Python
Pygame Event事件模块的详细示例
Nov 17 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代码审计比较有意思的例子
2014/05/07 PHP
php实现获取局域网所有用户的电脑IP和主机名、及mac地址完整实例
2014/07/18 PHP
PHP获取当前页面URL函数实例
2014/10/22 PHP
10个实用的PHP正则表达式汇总
2014/10/23 PHP
删除PHP数组中头部、尾部、任意元素的实现代码
2017/04/10 PHP
thinkPHP5框架设置404、403等http状态页面的方法
2018/06/05 PHP
Laravel5.1 框架路由基础详解
2020/01/04 PHP
JS 创建对象(常见的几种方法)
2008/11/03 Javascript
Javascript实现DIV滚动自动滚动到底部的代码
2012/03/01 Javascript
一个简单的网站访问JS计数器 刷新1次加1次访问
2012/09/20 Javascript
使用JS 清空File控件的路径值
2013/07/08 Javascript
js实现Select列表各项上移和下移的方法
2015/08/14 Javascript
微信js-sdk分享功能接口常用逻辑封装示例
2016/10/13 Javascript
原生js实现可爱糖果数字时间特效
2016/12/30 Javascript
bootstrap为水平排列的表单和内联表单设置可选的图标
2017/02/15 Javascript
template.js前端模板引擎使用详解
2017/10/10 Javascript
基于百度地图api清除指定覆盖物(Overlay)的方法
2018/01/26 Javascript
详解使用create-react-app添加css modules、sasss和antd
2018/07/31 Javascript
jquery简单实现纵向的无缝滚动代码实例
2019/04/01 jQuery
Layui 导航默认展开和菜单栏选中高亮设置的方法
2019/09/04 Javascript
layui自定义工具栏的方法
2019/09/19 Javascript
uni-app如何实现增量更新功能
2020/01/03 Javascript
JavaScript多种图形实现代码实例
2020/06/28 Javascript
JavaScript实现表单验证功能
2020/12/09 Javascript
vue 通过base64实现图片下载功能
2020/12/19 Vue.js
[03:03]DOTA2 2017国际邀请赛开幕战队入场仪式
2017/08/09 DOTA
举例讲解Python程序与系统shell交互的方式
2015/04/09 Python
Jupyter notebook在mac:linux上的配置和远程访问的方法
2019/01/14 Python
django执行数据库查询之后实现返回的结果集转json
2020/03/31 Python
台湾最大网路书店:博客来
2018/03/18 全球购物
荷兰手表网站:Watch2Day
2018/07/02 全球购物
怎样写好自我评价呢?
2014/02/16 职场文书
《三顾茅庐》教学反思
2014/04/10 职场文书
机械操作工岗位职责
2014/08/08 职场文书
你会写报告?产品体验报告到底该怎么写?
2019/08/14 职场文书
Go gRPC进阶教程gRPC转换HTTP
2022/06/16 Golang