with从Python 2.5就有,需要from __future__ import with_statement。自python 2.6开始,成为默认关键字。
也就是说with是一个控制流语句,跟if/for/while/try之类的是一类的,with可以用来简化try finally代码,看起来可以比try finally更清晰。
# @param Python with...as... 语法深入解析
# @author 编程之家 jb51.cc|www.jb51.cc
with EXPRESSION [ as VARIABLE] WITH-BLOCK
# End www.jb51.cc
基本思想是with所求值的对象必须有一个__enter__()方法,一个__exit__()方法。
紧跟with后面的语句被求值后,返回对象的__enter__()方法被调用,这个方法的返回值将被赋值给as后面的变量。当with后面的代码块全部被执行完之后,将调用前面返回对象的__exit__()方法。
with expresion as variable的执行过程是,首先执行__enter__函数,它的返回值会赋给as后面的variable,想让它返回什么就返回什么,只要你知道怎么处理就可以了,如果不写as variable,返回值会被忽略。
然后,开始执行with-block中的语句,不论成功失败(比如发生异常、错误,设置sys.exit()),在with-block执行完成后,会执行__exit__函数。
这样的过程其实等价于:
# @param Python with...as... 语法深入解析
# @author 编程之家 jb51.cc|www.jb51.cc
try:
执行 __enter__的内容
执行 with_block.
finally:
执行 __exit__内容
# End www.jb51.cc
再看个例子
# @param Python with...as... 语法深入解析
# @author 编程之家 jb51.cc|www.jb51.cc
file = open("/tmp/foo.txt")
try:
data = file.read()
finally:
file.close()
# End www.jb51.cc
# @param Python with...as... 语法深入解析
# @author 编程之家 jb51.cc|www.jb51.cc
with open("/tmp/foo.txt") as file:
data = file.read()
# End www.jb51.cc
# @param Python with...as... 语法深入解析
# @author 编程之家 jb51.cc|www.jb51.cc
#!/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
# End www.jb51.cc
执行结果为
# @param Python with...as... 语法深入解析
# @author 编程之家 jb51.cc|www.jb51.cc
In __enter__()
sample: Foo
In __exit__()
# End www.jb51.cc
1. __enter__()方法被执行
2. __enter__()方法返回的值 - 这个例子中是"Foo",赋值给变量'sample'
3. 执行代码块,打印变量"sample"的值为 "Foo"
4. __exit__()方法被调用with真正强大之处是它可以处理异常。可能你已经注意到Sample类的__exit__方法有三个参数- val,type 和 trace。这些参数在异常处理中相当有用。
原文链接:https://www.f2er.com/python/527045.html