理解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的curses库使用教程
Apr 11 Python
Python切片知识解析
Mar 06 Python
python实现决策树
Dec 21 Python
python opencv之分水岭算法示例
Feb 24 Python
零基础使用Python读写处理Excel表格的方法
May 02 Python
pyqt5让图片自适应QLabel大小上以及移除已显示的图片方法
Jun 21 Python
django自带调试服务器的使用详解
Aug 29 Python
python2与python3爬虫中get与post对比解析
Sep 18 Python
python之语音识别speech模块
Sep 09 Python
python和node.js生成当前时间戳的示例
Sep 29 Python
python神经网络编程之手写数字识别
May 08 Python
pytest实现多进程与多线程运行超好用的插件
Jul 15 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
Windows IIS PHP 5.2 安装与配置方法
2009/06/08 PHP
PHP 中 var_export、print_r、var_dump 调试中的区别
2018/06/19 PHP
PHP实现浏览器格式化显示XML的方法示例
2019/01/22 PHP
一个符号插入器 中用到的js代码
2007/09/04 Javascript
浅谈javascript六种数据类型以及特殊注意点
2013/12/20 Javascript
jquery实现对联广告的方法
2015/02/05 Javascript
jQuery实现手机号码输入提示功能实例
2015/04/30 Javascript
JavaScript中的parse()方法使用简介
2015/06/12 Javascript
jquery-tips悬浮提示插件分享
2015/07/31 Javascript
jquery简单实现带渐显效果的选项卡菜单代码
2015/09/01 Javascript
js判断出两个字符串最大子串的函数实现方法
2016/11/01 Javascript
从零开始学习搭建React脚手架项目
2018/08/23 Javascript
vue中前进刷新、后退缓存用户浏览数据和浏览位置的实例讲解
2018/09/21 Javascript
小程序调用微信支付的方法
2019/09/26 Javascript
Vue.js实现可编辑的表格
2019/12/11 Javascript
JavaScript设计模型Iterator实例解析
2020/01/22 Javascript
python生成指定尺寸缩略图的示例
2014/05/07 Python
Python Json模块中dumps、loads、dump、load函数介绍
2018/05/15 Python
Python中一些不为人知的基础技巧总结
2018/05/19 Python
pycharm修改文件的默认打开方式的步骤
2019/07/29 Python
python django model联合主键的例子
2019/08/06 Python
tensorflow:指定gpu 限制使用量百分比,设置最小使用量的实现
2020/02/06 Python
pytorch之Resize()函数具体使用详解
2020/02/27 Python
a标签下载链接的简单实现
2016/09/13 HTML / CSS
美国玛丽莎收藏奢华时尚商店:Marissa Collections
2016/11/21 全球购物
MADE法国:提供原创设计师家具
2018/09/18 全球购物
库存图片、照片、矢量图、视频和音乐:Shutterstock
2021/02/12 全球购物
外贸业务员求职自荐信分享
2013/09/21 职场文书
工商管理本科毕业生求职信范文
2013/10/05 职场文书
财务会计专业个人求职信范本
2014/01/08 职场文书
超级搞笑检讨书
2014/01/15 职场文书
国际贸易专业个人求职信格式
2014/02/02 职场文书
南京市纪委监察局整改方案
2014/09/16 职场文书
法务专员岗位职责
2015/02/14 职场文书
餐饮店长岗位职责
2015/04/14 职场文书
HTML中的表单Form实现居中效果
2021/05/25 HTML / CSS