理解Python中的With语句


Posted in Python onFebruary 02, 2015

With语句是什么?

有一些任务,可能事先需要设置,事后做清理工作。对于这种场景,Python的with语句提供了一种非常方便的处理方式。一个很好的例子是文件处理,你需要获取一个文件句柄,从文件中读取数据,然后关闭文件句柄。

如果不用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__()方法。

下面例子可以具体说明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__()

正如你看到的,

__enter__()方法被执行

__enter__()方法返回的值 - 这个例子中是"Foo",赋值给变量'sample'
执行代码块,打印变量"sample"的值为 "Foo"

__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后面的get_sample()变成了Sample()。这没有任何关系,只要紧跟with后面的语句所返回的对象有 __enter__()和__exit__()方法即可。此例中,Sample()的__enter__()方法返回新创建的Sample对象,并赋值给变量sample。

代码执行后:

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_somet hing()

  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语句是提供一个有效的机制,让代码更简练,同时在异常产生时,清理工作更简单。

示例代码可以在Github上面找到。

Python 相关文章推荐
Python数组遍历的简单实现方法小结
Apr 27 Python
创建pycharm的自定义python模板方法
May 23 Python
python 按不同维度求和,最值,均值的实例
Jun 28 Python
python调用百度语音识别实现大音频文件语音识别功能
Aug 30 Python
Python实现合并excel表格的方法分析
Apr 13 Python
GitHub 热门:Python 算法大全,Star 超过 2 万
Apr 29 Python
python实现函数极小值
Jul 10 Python
PyTorch之图像和Tensor填充的实例
Aug 18 Python
python图片指定区域替换img.paste函数的使用
Apr 09 Python
Django模板标签{% for %}循环,获取制定条数据实例
May 14 Python
10张动图学会python循环与递归问题
Feb 06 Python
python上下文管理器异常问题解决方法
Feb 07 Python
Linux环境下MySQL-python安装过程分享
Feb 02 #Python
Python中用pycurl监控http响应时间脚本分享
Feb 02 #Python
Python列表(list)常用操作方法小结
Feb 02 #Python
Python Sleep休眠函数使用简单实例
Feb 02 #Python
Python中实现从目录中过滤出指定文件类型的文件
Feb 02 #Python
Python实现二分法算法实例
Feb 02 #Python
Python标准异常和异常处理详解
Feb 02 #Python
You might like
php 数组排序 array_multisort与uasort的区别
2011/03/24 PHP
基于递归实现的php树形菜单代码
2014/11/19 PHP
简单介绍win7下搭建apache+php+mysql开发环境
2015/08/06 PHP
php基于jquery的ajax技术传递json数据简单实例
2016/04/15 PHP
PHP+Ajax 检测网络是否正常实例详解
2016/12/16 PHP
javascript之querySelector和querySelectorAll使用介绍
2011/12/20 Javascript
JavaScript Ajax Json实现上下级下拉框联动效果实例代码
2013/11/23 Javascript
js 设置缓存及获取设置的缓存
2014/05/08 Javascript
JQuery自适应窗口大小导航菜单附源码下载
2015/09/01 Javascript
简述jQuery ajax的执行顺序
2016/01/05 Javascript
jquery实现界面无刷新加载登陆注册
2016/07/30 Javascript
jQuery 局部div刷新和全局刷新方法总结
2016/10/05 Javascript
如何快速上手Vuex
2017/02/14 Javascript
angularjs ui-router中路由的二级嵌套
2017/03/10 Javascript
Vue.2.0.5实现Class 与 Style 绑定的实例
2017/06/20 Javascript
使用vue完成微信公众号网页小记(推荐)
2019/04/28 Javascript
NodeJs 实现简单WebSocket即时通讯的示例代码
2019/08/05 NodeJs
vscode调试node.js的实现方法
2020/03/22 Javascript
python的即时标记项目练习笔记
2014/09/18 Python
详解Python中的文本处理
2015/04/11 Python
python实现的系统实用log类实例
2015/06/30 Python
python 中if else 语句的作用及示例代码
2018/03/05 Python
Python正则表达式如何匹配中文
2020/05/27 Python
Pycharm打开已有项目配置python环境的方法
2020/07/03 Python
python使用requests库爬取拉勾网招聘信息的实现
2020/11/20 Python
Pycharm-community-2020.2.3 社区版安装教程图文详解
2020/12/08 Python
python正则表达式re.match()匹配多个字符方法的实现
2021/01/27 Python
css3中检验表单的required,focus,valid和invalid样式
2014/02/21 HTML / CSS
英语商务邀请函范文
2014/01/16 职场文书
语文教学感言
2014/02/06 职场文书
网吧最新创业计划书范文
2014/03/27 职场文书
船舶工程技术专业求职信
2014/08/07 职场文书
运动会跳远广播稿5篇
2014/09/17 职场文书
教师正风肃纪剖析材料
2014/10/20 职场文书
监守自盗观后感
2015/06/10 职场文书
《花钟》教学反思
2016/02/17 职场文书