29 | 巧用上下文管理器和With语句精简代码
下载APP
关闭
渠道合作
推荐作者
29 | 巧用上下文管理器和With语句精简代码
2019-07-15 景霄 来自北京
《Python核心技术与实战》
课程介绍
讲述:冯永吉
时长08:30大小7.79M
你好,我是景霄。
我想你对 Python 中的 with 语句一定不陌生,在专栏里它也曾多次出现,尤其是在文件的输入输出操作中,不过我想,大部分人可能习惯了它的使用,却并不知道隐藏在其背后的“秘密”。
那么,究竟 with 语句要怎么用,与之相关的上下文管理器(context manager)是什么,它们之间又有着怎样的联系呢?这节课,我就带你一起揭开它们的神秘面纱。
什么是上下文管理器?
在任何一门编程语言中,文件的输入输出、数据库的连接断开等,都是很常见的资源管理操作。但资源都是有限的,在写程序时,我们必须保证这些资源在使用过后得到释放,不然就容易造成资源泄露,轻者使得系统处理缓慢,重则会使系统崩溃。
光说这些概念,你可能体会不到这一点,我们可以看看下面的例子:
这里我们一共打开了 10000000 个文件,但是用完以后都没有关闭它们,如果你运行该段代码,便会报错:
这就是一个典型的资源泄露的例子。因为程序中同时打开了太多的文件,占据了太多的资源,造成系统崩溃。
为了解决这个问题,不同的编程语言都引入了不同的机制。而在 Python 中,对应的解决方式便是上下文管理器(context manager)。上下文管理器,能够帮助你自动分配并且释放资源,其中最典型的应用便是 with 语句。所以,上面代码的正确写法应该如下所示:
这样,我们每次打开文件“test.txt”,并写入‘hello’之后,这个文件便会自动关闭,相应的资源也可以得到释放,防止资源泄露。当然,with 语句的代码,也可以用下面的形式表示:
要注意的是,最后的 finally block 尤其重要,哪怕在写入文件时发生错误异常,它也可以保证该文件最终被关闭。不过与 with 语句相比,这样的代码就显得冗余了,并且还容易漏写,因此我们一般更倾向于使用 with 语句。
另外一个典型的例子,是 Python 中的 threading.lock 类。举个例子,比如我想要获取一个锁,执行相应的操作,完成后再释放,那么代码就可以写成下面这样:
而对应的 with 语句,同样非常简洁:
我们可以从这两个例子中看到,with 语句的使用,可以简化了代码,有效避免资源泄露的发生。
上下文管理器的实现
基于类的上下文管理器
了解了上下文管理的概念和优点后,下面我们就通过具体的例子,一起来看看上下文管理器的原理,搞清楚它的内部实现。这里,我自定义了一个上下文管理类 FileManager,模拟 Python 的打开、关闭文件操作:
需要注意的是,当我们用类来创建上下文管理器时,必须保证这个类包括方法”__enter__()”和方法“__exit__()”。其中,方法“__enter__()”返回需要被管理的资源,方法“__exit__()”里通常会存在一些释放、清理资源的操作,比如这个例子中的关闭文件等等。
而当我们用 with 语句,执行这个上下文管理器时:
下面这四步操作会依次发生:
方法“__init__()”被调用,程序初始化对象 FileManager,使得文件名(name)是"test.txt",文件模式 (mode) 是'w';
方法“__enter__()”被调用,文件“test.txt”以写入的模式被打开,并且返回 FileManager 对象赋予变量 f;
字符串“hello world”被写入文件“test.txt”;
方法“__exit__()”被调用,负责关闭之前打开的文件流。
因此,这个程序的输出是:
另外,值得一提的是,方法“__exit__()”中的参数“exc_type, exc_val, exc_tb”,分别表示 exception_type、exception_value 和 traceback。当我们执行含有上下文管理器的 with 语句时,如果有异常抛出,异常的信息就会包含在这三个变量中,传入方法“__exit__()”。
因此,如果你需要处理可能发生的异常,可以在“__exit__()”添加相应的代码,比如下面这样来写:
这里,我们在 with 语句中手动抛出了异常“exception raised”,你可以看到,“__exit__()”方法中异常,被顺利捕捉并进行了处理。不过需要注意的是,如果方法“__exit__()”没有返回 True,异常仍然会被抛出。因此,如果你确定异常已经被处理了,请在“__exit__()”的最后,加上“return True”这条语句。
同样的,数据库的连接操作,也常常用上下文管理器来表示,这里我给出了比较简化的代码:
与前面 FileManager 的例子类似:
方法“__init__()”负责对数据库进行初始化,也就是将主机名、接口(这里是 localhost 和 8080)分别赋予变量 hostname 和 port;
方法“__enter__()”连接数据库,并且返回对象 DBConnectionManager;
方法“__exit__()”则负责关闭数据库的连接。
这样一来,只要你写完了 DBconnectionManager 这个类,那么在程序每次连接数据库时,我们都只需要简单地调用 with 语句即可,并不需要关心数据库的关闭、异常等等,显然大大提高了开发的效率。
基于生成器的上下文管理器
诚然,基于类的上下文管理器,在 Python 中应用广泛,也是我们经常看到的形式,不过 Python 中的上下文管理器并不局限于此。除了基于类,它还可以基于生成器实现。接下来我们来看一个例子。
比如,你可以使用装饰器 contextlib.contextmanager,来定义自己所需的基于生成器的上下文管理器,用以支持 with 语句。还是拿前面的类上下文管理器 FileManager 来说,我们也可以用下面形式来表示:
这段代码中,函数 file_manager() 是一个生成器,当我们执行 with 语句时,便会打开文件,并返回文件对象 f;当 with 语句执行完后,finally block 中的关闭文件操作便会执行。
你可以看到,使用基于生成器的上下文管理器时,我们不再用定义“__enter__()”和“__exit__()”方法,但请务必加上装饰器 @contextmanager,这一点新手很容易疏忽。
讲完这两种不同原理的上下文管理器后,还需要强调的是,基于类的上下文管理器和基于生成器的上下文管理器,这两者在功能上是一致的。只不过,
基于类的上下文管理器更加 flexible,适用于大型的系统开发;
而基于生成器的上下文管理器更加方便、简洁,适用于中小型程序。
无论你使用哪一种,请不用忘记在方法“__exit__()”或者是 finally block 中释放资源,这一点尤其重要。
总结
这节课,我们先通过一个简单的例子,了解了资源泄露的易发生性,和其带来的严重后果,从而引入了应对方案——即上下文管理器的概念。上下文管理器,通常应用在文件的打开关闭和数据库的连接关闭等场景中,可以确保用过的资源得到迅速释放,有效提高了程序的安全性,
接着,我们通过自定义上下文管理的实例,了解了上下文管理工作的原理,并一起学习了基于类的上下文管理器和基于生成器的上下文管理器,这两者的功能相同,具体用哪个,取决于你的具体使用场景。
另外,上下文管理器通常和 with 语句一起使用,大大提高了程序的简洁度。需要注意的是,当我们用 with 语句执行上下文管理器的操作时,一旦有异常抛出,异常的类型、值等具体信息,都会通过参数传入“__exit__()”函数中。你可以自行定义相关的操作对异常进行处理,而处理完异常后,也别忘了加上“return True”这条语句,否则仍然会抛出异常。
思考题
那么,在你日常的学习工作中,哪些场景使用过上下文管理器?使用过程中又遇到了哪些问题,或是有什么新的发现呢?欢迎在下方留言与我讨论,也欢迎你把这篇文章分享出去,我们一起交流,一起进步。
分享给需要的人,Ta购买本课程,你将得18元
生成海报并分享
赞 26
提建议
© 版权归极客邦科技所有,未经许可不得传播售卖。 页面已增加防盗追踪,如有侵权极客邦将依法追究其法律责任。
上一篇
28 | 如何合理利用assert?
下一篇
30 | 真的有必要写单元测试吗?
精选留言(35)
- AllenGFLiu2019-07-15第一次有教程提到这个上下文管理器,学习。 对知识的学习就是需要从多角度重复去看,在这个过程中查遗补缺,才能保持不断进步。
作者回复: 嗯嗯,是这样的
共 2 条评论25 - ajodfaj2019-07-15with tf.Session() as sess共 1 条评论16
- enjoylearning2019-07-15主要用于数据库连接15
- new2019-07-15打开文件时用最方便13
- LJK2019-07-15老师好,请问基于类的上下文,“__enter__“方法什么时候返回self呢?DBConnectionManager的例子中可以说明一下为什么是返回self不是返回self.connection么?共 2 条评论8
- 安排2019-07-15是不是只有程序出了with代码块,管理的对象才会析构,也就是释放资源?
作者回复: 准确的说释放资源的行为发生在with语句的最后(上下文管理器的__exit__函数内)
共 2 条评论7 - Geek_59f23e2019-07-16UnboundLocalError: local variable 'f' referenced before assignment 最后一个例子有报错哦,基于生成器的上下文管理器那儿,提示说finally语句里的f变量没有先声明。共 1 条评论4
- 布凡2020-09-06这里with 的用法和C#中的using 一样,表示资源变量在一个块中有效,块结束后自动回收资源2
- 火锅小王子2019-11-02现在才学这个专栏 感觉落后了好多同学 想问下老师一个问题 这里讲到了上下文管理器 可不可以分析下async with异步上下文的逻辑和用法 ?一直不是很明白这个点 😀
作者回复: 后面有讲到
2 - Carl2019-10-21为什么平时使用 with open() as f 时可以畅通无阻呢? 是因为open这个函数在源码里就用@contextmanager装饰了吗?共 2 条评论2
- 瞳梦2019-07-17请问数据库的那个例子,__enter__()中返回self.connection是不是更符合实际应用中的情况。2
- fly10242019-08-03有个疑问: 对于file和Lock这两个使用with用法,python会自动调用对应释放资源的函数(close和release),这两个释放资源的函数也是定义__exit函数里的吗?希望老师解答一下。
作者回复: 对
1 - 张洪阆2019-07-28with限制了对象的作用域,相当于是个临时对象1
- Geek_d848f72019-07-15还是不怎么清楚基于生成器的上下文管理器的运行过程
作者回复: 哪里不明白?文章中应该讲的很清楚了
1 - 倾2019-07-15只是在文件操作时使用,今天第一次学习到还能这么用。1
- Fearless.2021-12-19yield f 这里用生成器的意义是啥呢?
- 王位庆2020-10-22最近正好看with语句,然后看课程学习了一下。基本了解了原理
- 郭纯2020-10-17我的理解在 __enter__ 和 __exit__ 魔术方法内定义资源初始化和资源释放的逻辑.也就是两个回调钩子. 当使用 with 的时候会在代码块内创建上下文.在上下文开始的时候执行 __enter__ 钩子函数. 在上下文销毁之前执行__exit__ 钩子函数. 提供了类似生命周期的机制.
- 追风筝的人2020-08-12这一节学到了with 的上下文管理器,之前都是f.close( )
- 风含叶2019-12-151. 主要是 io 操作,包含 文件打开关闭、网络请求