我们都知道打开文件有两种方法:
f = open() with open() as f:
这两种方法的区别就是第一种方法需要我们自己关闭文件;f.close(),而第二种方法不需要我们自己关闭文件,无论是否出现异常,with都会自动帮助我们关闭文件,这是为什么呢?
我们先自定义一个类,用with来打开它:
|----------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 | class Foo(): ``def __enter__(``self``): ``print``(``"enter called"``) ``def __exit__(``self``, exc_type, exc_val, exc_tb): ``print``(``"exit called"``) ``print``(``"exc_type :%s"``%``exc_type) ``print``(``"exc_val :%s"``%``exc_val) ``print``(``"exc_tb :%s"``%``exc_tb) with Foo() as foo: ``print``(``"hello python"``) ``a ``= 1``/``0 ``print``(``"hello end"``) |
执行结果:
|----------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| 1 2 3 4 5 6 7 8 9 10 11 12 | enter called Traceback (most recent call last): hello python exit called exc_type :<``class 'ZeroDivisionError'``> exc_val :division by zero ``File "F:/workspaces/python_workspaces/flask_study/with.py"``, line ``25``, ``in <module> ``a ``= 1``/``0 exc_tb :<traceback ``object at ``0x0000023C4EDBB9C8``> ZeroDivisionError: division by zero Process finished with exit code ``1 |
我们看到,执行结果的输入顺序,分析如下:
当我们with Foo() as foo:时,此时会执行__enter__方法,然后进入执行体,也就是:
|-------|----------------------------------------------------------------------------|
| 1 2 3 | print``(``"hello python"``) a ``= 1``/``0 print``(``"hello end"``) |
语句,但是在a=1/0出现了异常,with将会中止,此时就执行__exit__方法,就算不出现异常,当执行体被执行完毕之后,__exit__方法仍然被执行一次。
我们回到with open("file")as f: 不用关闭文件的原因就是在__exit__方法中,存在关闭文件的操作,所以不用我们手工关闭文件,with已将为我们做好了这个操作,这就可以理解了。