Python上下文管理器Content Manager


Posted in Python onJune 26, 2021

在 Python 中,我们会经常听到上下文管理器(Context Manager),那我们探讨下这是什么,又有什么功能。

在 Python 中的上下文管理器中,使用 with 打开文件是使用最多的,其中离开 with 包含的语句后会执行一些类似于清理的工作,如关闭文件,关闭连接对象等操作。

实践

我们在代码实践的时候,忽略了在同一代码片段中,先打开文件,然后直接对文件进行其他处理,因为这样没有任何意义,资源是处于被占用的情况。

先看下面检测的代码:

#!/usr/bin/env python
# _*_ coding: UTF-8 _*_
# MedusaSorcerer Script
import os


class OpenFile:
    def __init__(self):
        self.file = None

    def open(self, path):
        self.file = open(path, 'w')


if __name__ == '__main__':
    file_path = 'medusa.md'
    file = OpenFile()
    file.open(file_path)
    os.remove(file_path)

代码中我们把文件对象,进行了实例属性的方式引用,在此之后,我们使用 os 模块进行删除被写入的文件。执行改代码片段后,会出现以下内容:

Traceback (most recent call last):
  File "medusa/main.py", line 19, in <module>
    os.remove(file_path)
PermissionError: [WinError 32] 另一个程序正在使用此文件,进程无法访问。: 'medusa.md'

Process finished with exit code 1

那是因为被删除的文件没有得到资源释放。我们在上面的基础上进行套用函数的方式:

#!/usr/bin/env python
# _*_ coding: UTF-8 _*_
# MedusaSorcerer Script
import os


class OpenFile:
    def __init__(self):
        self.file = None

    def open(self, path):
        self.file = open(path, 'w')


def open_file(path):
    file = OpenFile()
    file.open(path)


if __name__ == '__main__':
    file_path = 'medusa.md'
    open_file(file_path)
    os.remove(file_path)

这段代码会成功的被执行成功,原因是当你执行函数的时候,函数内的临时变量将被回收释放,因此 OpenFile 的实例对象被释放了,实例属性也就不存在而被释放,所以会执行成功。

那是否我们的操作都应该使用函数包裹的方式执行呢?with 的出现,完美解决了这个问题:

#!/usr/bin/env python
# _*_ coding: UTF-8 _*_
# MedusaSorcerer Script
import os

if __name__ == '__main__':
    file_path = 'medusa.md'
    with open(file_path, 'w') as f:
        print(f)
    os.remove(file_path)

在 with 语法中,将后面打开文件的操作,返回的文件对象,赋值给 f 变量,在结构体中输出了 f 变量的内容,并且在结构体外删除了该文件:

medusa\python.exe medusa/main.py
<_io.TextIOWrapper name='medusa.md' mode='w' encoding='cp936'>

Process finished with exit code 0

在没有使用 close() 的情况下,依旧可以对文件进行删除,这就是上下文管理的美妙。

实现

上下文管理,实际上是实现了 __enter__ 和 __exit__ 方法:

#!/usr/bin/env python
# _*_ coding: UTF-8 _*_
# MedusaSorcerer Script


class Medusa:

    def __init__(self):
        print('__init__')

    def __enter__(self):
        print('__enter__')

    def __exit__(self, exc_type, exc_val, exc_tb):
        print('__exit__')


if __name__ == '__main__':
    medusa = Medusa()
    with medusa:
        print('with object')
    print('finish')

以下是输出结果:

__init__
__enter__
with object
__exit__
finish

我们发现魔法方法在结合某些语法后会发生自动调度,所以,上下文管理中就在自动调度中,关闭了某些对象。

优点

实现上下文管理可以简化我们的代码,让代码更加简单易读,使用最少的代码量,就可以完成全部工作。

到此这篇关于Python上下文管理器Content Manager的文章就介绍到这了,更多相关Python上下文管理器内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Python 相关文章推荐
使用PYTHON接收多播数据的代码
Mar 01 Python
python中__call__方法示例分析
Oct 11 Python
Python内置函数之filter map reduce介绍
Nov 30 Python
Python中使用Tkinter模块创建GUI程序实例
Jan 14 Python
在Python中通过threading模块定义和调用线程的方法
Jul 12 Python
Python利用matplotlib.pyplot绘图时如何设置坐标轴刻度
Apr 09 Python
如何利用Boost.Python实现Python C/C++混合编程详解
Nov 08 Python
Windows下Python3.6安装第三方模块的方法
Nov 22 Python
关于ResNeXt网络的pytorch实现
Jan 14 Python
pycharm全局搜索的具体步骤
Jul 28 Python
python3 googletrans超时报错问题及翻译工具优化方案 附源码
Dec 23 Python
详解NumPy中的线性关系与数据修剪压缩
May 25 Python
仅用几行Python代码就能复制她的U盘文件?
总结几个非常实用的Python库
Jun 26 #Python
手残删除python之后的补救方法
Python办公自动化之教你用Python批量识别发票并录入到Excel表格中
Python Pandas模块实现数据的统计分析的方法
Jun 24 #Python
FP-growth算法发现频繁项集——发现频繁项集
能让Python提速超40倍的神器Cython详解
Jun 24 #Python
You might like
PHP syntax error, unexpected $end 错误的一种原因及解决
2008/10/25 PHP
解析PHP跳出循环的方法以及continue、break、exit的区别介绍
2013/07/01 PHP
Laravel实现构造函数自动依赖注入的方法
2016/03/16 PHP
YII框架关联查询操作示例
2019/04/29 PHP
jquery BS,dialog控件自适应大小
2009/07/06 Javascript
jBox 2.3基于jquery的最新多功能对话框插件 常见使用问题解答
2011/11/10 Javascript
在图片上显示左右箭头类似翻页的代码
2013/03/04 Javascript
js添加table的行和列 具体实现方法
2013/07/22 Javascript
文本框(input)获取焦点(onfocus)时样式改变的示例代码
2014/01/10 Javascript
jQuery学习笔记之总体架构
2014/06/03 Javascript
node中socket.io的事件使用详解
2014/12/15 Javascript
使用console进行性能测试
2015/04/27 Javascript
JavaScript缓冲运动实现方法(2则示例)
2016/01/08 Javascript
JavaScript Date对象详解
2016/03/01 Javascript
JavaScript 2048 游戏实例代码(简单易懂)
2016/03/25 Javascript
AngularJs基本特性解析(一)
2016/07/21 Javascript
js 简易版滚动条实例(适用于移动端H5开发)
2017/06/26 Javascript
Javascript(es2016) import和require用法和区别详解
2017/08/11 Javascript
jQuery实现table中两列CheckBox只能选中一个的示例
2017/09/22 jQuery
原生JS封装_new函数实现new关键字的功能
2018/08/12 Javascript
对于Python中RawString的理解介绍
2016/07/07 Python
Python实现爬虫爬取NBA数据功能示例
2018/05/28 Python
解决python3 pika之连接断开的问题
2018/12/18 Python
python 自定义装饰器实例详解
2019/07/20 Python
Python如何实现在字符串里嵌入双引号或者单引号
2020/03/02 Python
Cocopanda波兰:购买化妆品、护肤品、护发和香水
2020/05/25 全球购物
劳动模范事迹材料
2014/01/19 职场文书
护士节演讲稿开场白
2014/08/25 职场文书
领导班子作风建设剖析材料
2014/10/11 职场文书
骨干教师申报材料
2014/12/17 职场文书
三年级学生评语大全
2014/12/26 职场文书
故意伤害罪辩护词
2015/05/21 职场文书
美丽心灵观后感
2015/06/01 职场文书
盲山观后感
2015/06/11 职场文书
小学大队长竞选稿
2015/11/20 职场文书
导游词之云南-元阳梯田
2019/10/08 职场文书