data:image/s3,"s3://crabby-images/5f97a/5f97a51a9fad94561999d7c5f8d20e2c4f970fbd" alt="自学Python:编程基础、科学计算及数据分析"
3.4 上下文管理器与with语句
之前在介绍文件读写的时候我们说过,当一个文件打开的时候,可能会遇到不正常关闭的问题,这样可能会影响我们读写文件。
不仅是文件,当我们处理资源的时候都可能会遇到这样的问题,这些资源包括文件、线程、数据库、网络连接等等。
写入文件时,如果文件没有被正常关闭,会导致某些内容没有来得及写入的情况。
例如,用一个循环向文件中写入数据:
data:image/s3,"s3://crabby-images/28cee/28ceef8f0ce0a692c9eb0b846068162f2a74aba7" alt=""
当i循环到500的时候抛出了一个ZeroDivisionError,程序中断,上一行应该执行了写入“line 500”的操作。
打开这个文件,我们可能在文件结尾会看到类似如下的情况:
data:image/s3,"s3://crabby-images/ab9ae/ab9ae78377e4b40fb0f56afb586d9d2b5bbcdfd3" alt=""
data:image/s3,"s3://crabby-images/a9bb0/a9bb0294d5d8a393ab11f9141aa05b34e29eeebf" alt=""
对于上面的情况,可以采用try块的方式,在finally中确保文件f被正确关闭:
data:image/s3,"s3://crabby-images/af3c9/af3c9b6908237eab274d1737c65f7f7f206bb728" alt=""
finally能够保证f.close()正常执行。
打开文件会发现,文件的结尾保存的是正常的结果:
data:image/s3,"s3://crabby-images/fab19/fab19618a05af504e1abcb9a5b164d0be75ed12c" alt=""
资源使用的问题十分常见,都使用try块处理显得不够简洁。为此,Python提供了上下文管理器的机制来解决这个问题,它通常与关键字with一起使用。
对于上面的例子,我们用with语句调用的方式为:
data:image/s3,"s3://crabby-images/28e06/28e0697054ccf4f45aa3a88b76e7221f65eac3f5" alt=""
这与使用try块的效果相同,但是简洁了许多。
3.4.1 上下文管理器的原理
1. 基本形式和with语句
with语句的基本用法如下:
data:image/s3,"s3://crabby-images/33fb1/33fb1462e3bdb5ef2278b4eb90e642790fe1c04c" alt=""
其中<expression>是一个上下文管理器。
上下文管理器(Context Manager)是一个实现了.__enter__()方法和.__exit__()方法的对象。
文件对象包含这两个方法,所以是一个合法的上下文管理器,可以用在with语句中:
data:image/s3,"s3://crabby-images/667aa/667aae3a6203b991c65bd11578baf6448bc899cf" alt=""
在with语句中,上下文管理器的.__enter__()方法会在<statements>执行前执行,而.__exit__()方法会在<statements>执行结束后执行。
任何实现了这两种方法的对象都是一个合法的上下文管理器。不过,上下文管理器的.__exit__()方法需要接受3个额外参数(加上self是四个)。
例如,我们定义这样一个上下文管理器:
data:image/s3,"s3://crabby-images/4bc04/4bc04c88b3435683e1094691d3371cf32d0700ce" alt=""
使用这个管理器:
data:image/s3,"s3://crabby-images/534f3/534f3c164a4e13d5fa8f725b88e64ace2eccbffe" alt=""
如果<statements>在执行过程中抛出了异常,.__exit__()方法会先被执行,然后抛出异常:
data:image/s3,"s3://crabby-images/6c91d/6c91d3388e186ff79eca5d0144b1e3662b47453f" alt=""
data:image/s3,"s3://crabby-images/35247/35247cfb801a32623d8f6160e4bda8dfe4afb068" alt=""
2. 方法.__enter__()的返回值
为了在<statements>中使用文件对象,我们使用了as关键字的形式,将open()函数返回的文件对象赋给了f。事实上,as关键字只是将上下文管理器.__enter__()方法的返回值赋给了f,而文件对象的.__enter__()方法的返回值刚好是它本身:
data:image/s3,"s3://crabby-images/5375e/5375e925955b5239fa91af7dc4274c6b29b0d482" alt=""
我们修改TestManager的定义,将.__enter__()的返回值修改为字符串"My value!":
data:image/s3,"s3://crabby-images/0fb99/0fb99a6ea795279cc26e08a576fd4eb6e4787ee5" alt=""
然后用as关键字得到这个返回值:
data:image/s3,"s3://crabby-images/30dd3/30dd35eaddb70989f243a8aaf2ca6165df54100d" alt=""
返回这个上下文管理器本身是一种常用的设计:
data:image/s3,"s3://crabby-images/28152/2815222fbdceb0a9308ef21550fcb3c235bed8cd" alt=""
3. 方法.__exit__()与异常处理
在定义上下文管理器时,方法.__exit__()需要接受额外的参数,这些额外参数与异常处理相关。
重新定义TestManager,将这些参数打印出来:
data:image/s3,"s3://crabby-images/13128/13128e22f43f101220125746c8ee312f9c7e2e15" alt=""
没有异常时:
data:image/s3,"s3://crabby-images/1ba22/1ba22aa9e50bedb38df9b46c933aa5467b96c995" alt=""
当运行过程中抛出异常时:
data:image/s3,"s3://crabby-images/f9ebd/f9ebd9e0e43e43f32faa090e54a7bbab555195ce" alt=""
当运行出现异常时,这三个参数包含的是异常的具体信息。
在上面的例子中,我们只是简单的显示了异常的值,并没有像try-except块中的except块部分一样对异常进行处理,所以异常在执行完.__exit__()方法后被继续抛出了。
如果不想让异常继续抛出,我们只需要将.__exit__()方法的返回值设为True:
data:image/s3,"s3://crabby-images/b4de3/b4de30f3a1b077e44742c65e8a9c30cb9d9ac8b1" alt=""
3.4.2 模块contextlib
Python提供了contextlib模块来方便我们使用上下文管理器。
1.contextmanager装饰器
contextlib模块提供了装饰器contextmanager来实现一个简单的“上下文管理器”:
data:image/s3,"s3://crabby-images/cdb1d/cdb1d08cb860f448ea8e8d5f4667bb572241e8e3" alt=""
contextmanager作为一个装饰器,对所修饰的函数有固定的要求:
该函数必须是一个生成器,且yield只被执行一次。在该函数中,yield之前的部分可以看成是.__enter__()方法的部分,yield返回的值可以看成是.__enter__()方法返回的值,yield之后的部分可以看成是.__exit__()方法的部分。
函数test_manager()返回了一个上下文管理器:
data:image/s3,"s3://crabby-images/3aea7/3aea756d2b1f7ae91620bdba318c9eae7dad3f20" alt=""
使用yield的返回值:
data:image/s3,"s3://crabby-images/33863/3386375e889b56538c2f919b6e5a09653503e4e1" alt=""
不过,这样定义出来的“上下文管理器”并不能在出错的时候保证后续处理被执行:
data:image/s3,"s3://crabby-images/c1de0/c1de08bd53fddd7b44f4216df67fb0f8f34fdd54" alt=""
使用contextmanager构造的上下文管理器时,如果抛出异常,那么这个异常会在yield的地方重新被抛出,我们可以使用try块的形式对yield的部分进行处理。此外,为了保证对应.__exit__()的部分始终被执行,我们需要将yield后面的部分放入finally块中:
data:image/s3,"s3://crabby-images/ab8bc/ab8bcf615dff56720cbf54b965d79b19613977cb" alt=""
2.closing函数
contextlib模块中常用的还有closing()函数。该函数接受一个对象,返回一个确保该对象的.close()方法被调用的上下文管理器,它相当于这样的一个函数:
data:image/s3,"s3://crabby-images/2877c/2877cec275e231c3c971ae0616de6543b6456adc" alt=""
使用这种结构,我们能确保<object>的.close()方法最终被调用了,比如打开的网页:
data:image/s3,"s3://crabby-images/575cc/575cceeab24988a35df7877d13d2b9679eff0182" alt=""