with从Python 2.5就有,需要from __future__ import with_statement。自python 2.6开始,成为默认要害字。
也就是说with是一个节制流语句,跟if/for/while/try之类的是一类的,with可以用来简化try finally代码,看起来可以比try finally更清晰。
with EXPRESSION [ as VARIABLE] WITH-BLOCK
根基思想是with所求值的工具必需有一个__enter__()要领,一个__exit__()要领。
紧跟with后头的语句被求值后,返回工具的__enter__()要领被挪用,这个要领的返回值将被赋值给as后头的变量。当with后头的代码块全部被执行完之后,将挪用前面返回工具的__exit__()要领。
with expresion as variable的执行进程是,首先执行__enter__函数,它的返回值会赋给as后头的variable,想让它返回什么就返回什么,只要你知道怎么处理惩罚就可以了,假如不写as variable,返回值会被忽略。
然后,开始执行with-block中的语句,岂论乐成失败(好比产生异常、错误,配置sys.exit()),在with-block执行完成后,会执行__exit__函数。
这样的进程其实等价于:
try: 执行 __enter__的内容 执行 with_block. finally: 执行 __exit__内容
再看个例子
file = open("/tmp/foo.txt") try: data = file.read() finally: file.close()
利用with…as…的方法替换,修改后的代码是:
with open("/tmp/foo.txt") as file: data = file.read()
#!/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
执行功效为
In __enter__() sample: Foo In __exit__()
1. __enter__()要领被执行
2. __enter__()要领返回的值 – 这个例子中是"Foo",赋值给变量'sample'
3. 执行代码块,打印变量"sample"的值为 "Foo"
4. __exit__()要领被挪用with真正强大之处是它可以处理惩罚异常。大概你已经留意到Sample类的__exit__要领有三个参数- val, type 和 trace。这些参数在异常处理惩罚中相当有用。