理解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中用于转换字母为小写的lower()方法使用简介
May 19 Python
python中string模块各属性以及函数的用法介绍
May 30 Python
Python利用matplotlib生成图片背景及图例透明的效果
Apr 27 Python
Python实现小数转化为百分数的格式化输出方法示例
Sep 20 Python
Python3 循环语句(for、while、break、range等)
Nov 20 Python
tensorflow中next_batch的具体使用
Feb 02 Python
python2 与 pyhton3的输入语句写法小结
Sep 10 Python
详解Python logging调用Logger.info方法的处理过程
Feb 12 Python
python利用JMeter测试Tornado的多线程
Jan 12 Python
Python函数式编程实例详解
Jan 17 Python
python将音频进行变速的操作方法
Apr 08 Python
Python基础之元类详解
Apr 29 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错误信息方法的详解
2013/06/09 PHP
PHP设计模式之代理模式的深入解析
2013/06/13 PHP
用YUI做了个标签浏览效果
2007/02/20 Javascript
Mootools 1.2教程 输入过滤第一部分(数字)
2009/09/15 Javascript
Javascript var变量隐式声明方法
2009/10/19 Javascript
深入理解JavaScript作用域和作用域链
2011/10/21 Javascript
js+数组实现网页上显示时间/星期几的实用方法
2013/01/18 Javascript
js实现屏蔽默认快捷键调用自定义事件示例
2013/06/18 Javascript
javascript实现点击提交按钮后显示loading的方法
2015/07/03 Javascript
使用Bootstrap框架制作查询页面的界面实例代码
2016/05/27 Javascript
JavaScript常用正则函数用法示例
2017/01/23 Javascript
JS请求servlet功能示例
2017/06/01 Javascript
nodejs实现套接字服务功能详解
2018/06/21 NodeJs
利用不到200行代码写一款属于你自己的js类库
2019/07/08 Javascript
使用Bootstrap做一个朝代历史表
2019/12/10 Javascript
jQuery开发仿QQ版音乐播放器
2020/07/10 jQuery
JavaScript 中的六种循环方法
2021/01/06 Javascript
[56:35]DOTA2上海特级锦标赛C组小组赛#1 OG VS Archon第二局
2016/02/27 DOTA
python 字符串转列表 list 出现\ufeff的解决方法
2017/06/22 Python
Python从ZabbixAPI获取信息及实现Zabbix-API 监控的方法
2018/09/17 Python
python输入整条数据分割存入数组的方法
2018/11/13 Python
用python拟合等角螺线的实现示例
2019/12/27 Python
django ListView的使用 ListView中获取url中的参数值方式
2020/03/27 Python
python中的对数log函数表示及用法
2020/12/09 Python
HTML5地理定位实例
2014/10/15 HTML / CSS
英国探险旅游专家:Explore
2018/12/20 全球购物
华为python面试题
2016/05/03 面试题
探矿工程师自荐信
2014/01/24 职场文书
《小小雨点》教学反思
2014/02/18 职场文书
我的老师教学反思
2014/05/01 职场文书
创业培训计划书
2014/05/03 职场文书
廉政文化进校园广播稿
2014/10/20 职场文书
2015年公务员转正工作总结
2015/04/24 职场文书
《女娲补天》教学反思
2016/02/20 职场文书
Redis源码阅读:Redis字符串SDS详解
2021/07/15 Redis
css3属性选择器 “~”(波浪号) “,”(逗号) “+”(加号)和 “>”(大于号)
2022/04/19 HTML / CSS