一文带你搞懂Python上下文管理器
作者:曲鸟 发布时间:2022-11-26 05:15:50
一、什么是上下文管理器
我们在处理文件的时候经常看到下面这样的代码,它即是上下文管理器:
with open('test.txt', encoding='utf-8') as f:
print(f.readlines())
它的含义是打开当前目录下的test.txt文件并打印它里面的内容,与下面的代码效果是一样的:
f = open('test.txt', encoding='utf-8')
print(f.readlines())
f.close()
对比两种写法能够发现,使用with自动执行了f.close()(关闭文件)的这步操作,能够少写一点代码。
那这样的上下文管理器是怎么实现的,下面为你讲解。
二、如何实现上下文管理器
1. 通过类实现
如果要实现上面open
的上下文管理器功能,我们可以通过创建一个类,并添加__enter__
和__exit__
方法即可,如下面的代码所示:
class DiyOpen(object):
def __init__(self, filename, **kwargs):
self.f = open(filename, **kwargs)
def __enter__(self):
return self.f
def __exit__(self, exc_type, exc_val, exc_tb):
print('关闭文件')
self.f.close()
with DiyOpen('test.txt', encoding='utf-8') as f:
print(f.readlines())
输出结果
['第一行\n', '第二行\n', '第三行']
关闭文件
可以看到在我们打印出文件内容后,自动执行了关闭文件的操作。
那__enter__
和__exit__
的含义是什么,__exit__
后面的exc_type
, exc_val
, exc_tb
又是什么意思呢?
1)_enter_
__enter__
相对来说好理解的多,当出现with语句时,它就会被触发,有返回值时,会把返回值赋值给as声明的变量,也就是我们上面的as f中的f。
2)_exit_
__exit__
是在with执行完成后自动执行的,他后面的参数含义如下:
exc_type
:异常类型exc_val
:异常原因exc_tb
:堆栈追踪信息
当with中执行的代码报错时,除了不继续执行with包含的代码外,还会将报错信息放入上面的三个参数中,例如下面的代码:
class DiyOpen(object):
def __init__(self, filename, **kwargs):
self.f = open(filename, **kwargs)
def __enter__(self):
return self.f
def __exit__(self, exc_type, exc_val, exc_tb):
print(exc_type)
print(exc_val)
print(exc_tb)
self.f.close()
with DiyOpen('test.txt', encoding='utf-8') as f:
print(f.no())
输出结果
<class 'AttributeError'>
'_io.TextIOWrapper' object has no attribute 'no'
<traceback object at 0x000002A34B834900>
需要注意的是:
我们可以手动指定__exit__的返回值为True让它不报错。
没有异常信息时,上面的三个参数值都会为None
2. 通过contextlib实现
Python内置了contextlib
这个模块用于实现上下文管理器,它是通过生成器yield实现的,这个模块让我们不必再创建类和__enter__和__exit__了。
通过contextlib
实现open功能的代码如下:
from contextlib import contextmanager
@contextmanager
def diy_open(filename, **kwargs):
f = open(filename, **kwargs) # __init__
try:
yield f # __enter__
finally: # __exit__
f.close()
with diy_open('test.txt', encoding='utf-8') as f:
print(f.readlines())
总结
本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注脚本之家的更多内容!
来源:https://blog.csdn.net/momoda118/article/details/121988918
猜你喜欢
- python版本:3.8class object: """ The most base type "
- MySQL事务处理(TransAction)思考了很久,决定写一篇关于mysql事务(transaction)的博客,一来嘛,因为最近在复习
- 记录一下安装win10+GeForce GTX1060+CUDA 9.0+cuDNN7.3+tensorflow-gpu 1.12.0+py
- 1、利用php gd库的函数绘制3D扇形统计图<?phpheader("content-type","t
- HTML:<%-- 右键菜单 --%><div id="zTreeRightMenuContainer"
- 我的数据库和报表服务的版本如下:数据库:SQL Server 2008 R2报表服务:SQL Server 2008 R2 Reportin
- 需求:启动程序后,让用户输入工资,然后打印商品列表允许用户根据商品编号购买商品用户选择商品后,检测余额是否够,够就直接扣款,不够就提醒可随时
- 码代码时,有时候需要根据比较大小分别赋值:import randomseq = [random.randint(0, 1000) for _
- 前言和Word、Excel承载数据的能力相比,PPT的应用重点在于表演。比如一场发布会、一场演说、一次产品展示、一次客户沟通&hel
- 概述Golang 是一个跨平台的新生编程语言. 今天小白就带大家一起携手走进 Golang 的世界.数据类型转换数据类型转换可以帮助我们将一
- 删除备份和还原历史记录表中所有早于 oldest_date 的备份集的项目。由于执行备份或还原操作时会在备份和还原历史记录表中添加行,sp_
- 如下所示:# requests模块来请求页面# lxml模块的html构建selector选择器(格式化响应response)# from
- 有效地加载数据有时我们需大量地把数据加载到数据表,采用批量加载的方式比一个一个记录加载效率高,因为MySQL不用每加载一条记录就刷新一次索引
- python代码有一行标黄鼠标放在标黄的代码行会显示This code is unreachable(此代码无法访问)def do_teln
- python使用ctypes调用C/C++1. ctpes介绍ctypes is a foreign function library fo
- Python中对信号处理的模块主要是使用signal模块,但signal主要是针对Unix系统,所以在Windows平台上Python不能很
- 本文实例为大家分享了tensorflow实现弹性网络回归算法,供大家参考,具体内容如下python代码:#用tensorflow实现弹性网络
- 该脚本是为了结合之前的编写的脚本,来实现数据的比对模块,实现数据的自动化!由于数据格式是定死的,该代码只做参考,有什么问题可以私信我!CSV
- 前言最近参加了datawhale的组队学习活动,在组队学习动员下,开始通过强迫自己输出来实现更好的输入与处理,6-15开始自己的第一次文章发
- 本文实例为大家分享了python实现贪吃蛇的具体代码,供大家参考,具体内容如下import pygameimport sysimport r