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在windows命令行下输出彩色文字的方法
Mar 19 Python
Python中使用语句导入模块或包的机制研究
Mar 30 Python
python实现决策树C4.5算法详解(在ID3基础上改进)
May 31 Python
python 中if else 语句的作用及示例代码
Mar 05 Python
python3如何将docx转换成pdf文件
Mar 23 Python
详谈python3 numpy-loadtxt的编码问题
Apr 29 Python
对dataframe进行列相加,行相加的实例
Jun 08 Python
python 寻找list中最大元素对应的索引方法
Jun 28 Python
numpy数组广播的机制
Jul 12 Python
你还在@微信官方?聊聊Python生成你想要的微信头像
Sep 25 Python
python FTP批量下载/删除/上传实例
Dec 22 Python
将pycharm配置为matlab或者spyder的用法说明
Jun 08 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实例分享之通过递归实现删除目录下的所有文件详解
2014/05/15 PHP
php获取textarea的值并处理回车换行的方法
2014/10/20 PHP
php探针不显示内存解决方法
2019/09/17 PHP
Javascript学习笔记8 用JSON做原型
2010/01/11 Javascript
javascript textarea光标定位方法(兼容IE和FF)
2011/03/12 Javascript
Struts2的s:radio标签使用及用jquery添加change事件
2013/04/08 Javascript
js实现完全自定义可带多级目录的网页鼠标右键菜单方法
2015/02/28 Javascript
JavaScript将字符串转换为整数的方法
2015/04/14 Javascript
基于jQuery仿淘宝产品图片放大镜代码分享
2020/06/23 Javascript
浅谈js中的三种继承方式及其优缺点
2016/08/10 Javascript
bootstrap中使用google prettify让代码高亮的方法
2016/10/21 Javascript
探索Javascript中this的奥秘
2016/12/11 Javascript
JS简单判断函数是否存在的方法
2017/02/13 Javascript
JavaScript判断输入是否为数字类型的方法总结
2017/09/28 Javascript
js中int和string数据类型互相转化实例
2019/01/16 Javascript
总结4个方面优化Vue项目
2019/02/11 Javascript
bootstrap datepicker的基本使用教程
2019/07/09 Javascript
Angular处理未可知异常错误的方法详解
2021/01/17 Javascript
Vue多选列表组件深入详解
2021/03/02 Vue.js
为什么入门大数据选择Python而不是Java?
2018/03/07 Python
对Python使用mfcc的两种方式详解
2019/01/09 Python
python 多线程串行和并行的实例
2019/02/22 Python
Python多线程threading模块用法实例分析
2019/05/22 Python
全面介绍python中很常用的单元测试框架unitest
2020/12/14 Python
英国和爱尔兰最大的地毯零售商:Kukoon
2018/12/17 全球购物
工程业务员岗位职责
2013/12/31 职场文书
医学生毕业自我鉴定
2014/03/26 职场文书
中文专业自荐书
2014/06/29 职场文书
2014年化验室工作总结
2014/11/21 职场文书
学习焦裕禄观后感
2015/06/09 职场文书
餐馆开业致辞
2015/08/01 职场文书
2019年教师节活动策划方案
2019/09/09 职场文书
python模块与C和C++动态库相互调用实现过程示例
2021/11/02 Python
Java 通过手写分布式雪花SnowFlake生成ID方法详解
2022/04/07 Java/Android
在 Python 中利用 Pool 进行多线程
2022/04/24 Python
Java界面编程实现界面跳转
2022/06/16 Java/Android