懒人在思考 2018-02-17
以下为转载https://www.cnblogs.com/DswCnblog/p/6126588.html
with。。as。。一个使用场景是文件处理,你需要获取一个文件句柄,从文件中读取数据,然后关闭文件句柄。
普通的文件处理如下:
file = open("/tmp/foo.txt") data = file.read() file.close()#文件使用完毕后必须关闭,因为文件对象会占用操作系统的资源,并且操作系统同一时间能打开的文件数量也是有限的
由于文件读写时都有可能产生IOError,一旦出错,后面的f.close()就不会调用。所以,为了保证无论是否出错都能正确地关闭文件,我们可以使用try ... finally来实现:
file = open("/tmp/foo.txt") try: data = file.read() finally: file.close()
但是还有一个更简洁的实现,就是用with as
with open("/tmp/foo.txt") as file: data = file.read()
with所求值的对象必须有一个__enter__()方法,一个__exit__()方法。
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__()方法被执行,返回的对象赋值给变量'sample'
3. 执行代码块,打印变量"sample"的值为 "Foo"
4. __exit__()方法被调用
with真正强大之处是它可以处理异常。可能你已经注意到Sample类的__exit__方法有三个参数- val, type 和 trace。 这些参数在异常处理中相当有用。我们来改一下代码,看看具体如何工作的。
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/return bar + 10 with Sample() as sample: sample.do_something()
这个例子中,with后面的get_sample()变成了Sample()。这没有任何关系,只要紧跟with后面的语句所返回的对象有__enter__()和__exit__()方法即可。此例中,Sample()的__enter__()方法返回新创建的Sample对象,并赋值给变量sample。
代码执行后:
1 bash-3.2$ ./with_example02.py 2 type: <type 'exceptions.ZeroDivisionError'> 3 value: integer division or modulo by zero 4 trace: <traceback object at 0x1004a8128> 5 Traceback (most recent call last): 6 File "./with_example02.py", line 19, in <module> 7 sample.do_something() 8 File "./with_example02.py", line 15, in do_something 9 bar = 1/0 10 ZeroDivisionError: integer division or modulo by zero
实际上,在with后面的代码块抛出任何异常时,__exit__()方法被执行。正如例子所示,异常抛出时,与之关联的type,value和stack trace传给__exit__()方法,因此抛出的ZeroDivisionError异常被打印出来了。开发库时,清理资源,关闭文件等等操作,都可以放在__exit__方法当中。
因此,Python的with语句是提供一个有效的机制,让代码更简练,同时在异常产生时,清理工作更简单。