我们都知道打开文件有两种方法:
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已将为我们做好了这个操作,这就可以理解了。