Python之鸭子类型:一次搞懂with与上下文装饰器
wptr33 2025-05-11 01:42 24 浏览
引言
在鸭子类型的理念的基础之上,从关注类型,转变到关注特性和行为。结合Python中的魔法函数的体系,我们可以将自定义的类型,像内置类型一样被使用。今天这篇文章中,接着该话题,继续聊一下with语法块,以及上下文管理器的概念。
本文的主要内容有:
1、什么是上下文管理器
2、with语句
3、利用魔法函数自定义上下文管理器
什么是上下文管理器
所谓“上下文管理器(Context Manager)”在Python中是一种用于管理资源的对象,其中定义了进入上下文和退出上下文的行为。
通过实现__enter__()和__exit__()两个魔术方法,上下文管理器可以更加简洁、安全地进行资源的管理。上下文管理器可以确保资源在使用完毕后被正确地释放,即便发生异常。
这种机制可以有效降低资源管理的复杂度,尤其是能够有效避免资源泄露和处理资源管理中的复杂异常场景等。
学习一个特性,首先必须要问的,一定是学这个东西有什么用。所以,接下来简单列举几个上下文管理器的使用场景:
1、文件操作:简化文件的打开、读写、关闭的操作流程,降低异常处理的复杂度,确保文件使用完毕后总能被正确关闭。
2、数据库连接:数据库连接也是比较昂贵的资源开销,上下文管理器同样可以确保数据库连接在使用完毕后被正确关闭,从而释放资源。
3、线程锁:在并发编程中,锁的获取和释放是很关键的,如果由于异常,导致锁未能正常释放,可能导致死锁或者无限期等待的发生。上下文管理器同样可以保证锁资源在使用完成后被正确地释放。
4、网络连接:网络连接需要在使用完毕后被关闭,以释放资源。
5、临时文件:有些场景中,需要创建临时文件,并在使用完成后,自动删除它,这时,也可以考虑使用上下文管理器,确保临时文件的正确删除。
6、计时器:通过上下文管理器,我们也可以实现代码执行时间的统计需求。
with语句
with语句的作用是简化上下文管理器的使用,确保资源能够被正确地管理和释放。它可以避免常见的资源管理的问题,比如由于异常导致的文件未能关闭或者锁资源未能正常释放等。
with语句的作用机制,是要集合上下文管理器的两个魔术方法来实现的。前面我们已经提及过,所有的魔术方法,本质上不需要我们手动调用,Python解释器在特定的场景中会自动调用相应的魔术方法。
简单描述一下with语句块的作用机制:
1、Python在执行到with语句时,会自动调用该上下文管理器的__enter__()方法,该方法通常用于资源的分配或者初始化。返回的对象会绑定到with语句的as子句后面的变量(如果有的话)。
2、当要离开with语句块时,Python解释器会自动调用该上下文管理器的__exit__()方法,该方法通常用于清理资源或者处理异常。如果在with语句块中发生了异常,这些异常信息也会传递给__exit__()方法。
利用魔法函数自定义上下文管理器
在鸭子类型理念的指导下,定义上下文管理器变得非常简单。只要一个对象具备了进入上下文的相应处理行为,以及退出上下文的相应处理行为。那么这个对象,就是一个上下文管理器。
接下来,结合前面列举的两个场景,以实际代码演示自定义上下文管理器:
1、临时文件
import os
class TempFile:
def __init__(self, persist=False):
self.persist = persist
def __enter__(self):
self.path = f'{hex(id(self))}.tmp'
print(f'进入临时文件上下文,开始创建临时文件{self.path}')
self.file = open(self.path, 'a+')
return self
def __exit__(self, exc_type, exc_val, exc_tb):
print('准备退出临时文件上下文,开始清理资源')
self.file.close()
if not self.persist:
os.remove(self.path)
print(f'临时文件{self.path}被删除')
def write(self, content):
self.file.write(content)
def read(self):
return self.file.read()
def seek(self, offset):
self.file.seek(offset)
if __name__ == '__main__':
# 不保存的临时文件
with TempFile() as tf:
tf.write('hello python')
tf.seek(0)
print(tf.read())
# 保存的临时文件
with TempFile(persist=True) as tf:
tf.write('hello python')
tf.seek(0)
print(tf.read())
执行结果:
执行完成后,会发现第二个临时文件被保存了下来。
其实,Python中有现成的临时文件内置模块:tempfile,没必要我们自己来定义,通过代码简单演示一下:
import tempfile
# 删除的临时文件
with tempfile.NamedTemporaryFile('a+', delete=True) as tf:
tf.write('hello python')
tf.seek(0)
print(tf.read())
# 保留的临时文件
with tempfile.NamedTemporaryFile('a+', delete=False) as tf:
tf.write('hello python')
tf.seek(0)
print(tf.file.name)
print(tf.read())
执行结果:
2、代码执行时间统计
import time
class Timer:
def __enter__(self):
print('开始执行')
self.start = time.time()
def __exit__(self, exc_type, exc_val, exc_tb):
self.end = time.time()
print('结束执行')
print(f'总计耗时:{self.end - self.start:.2f}秒')
with Timer():
print('假装在认真工作')
time.sleep(2)
执行结果:
当然,同样的效果,也可以通过装饰器实现的,前面已经介绍过,这里就不再展开了。
总结
本文继续就鸭子类型与魔术方法的内容,进行展开介绍。首先介绍了上下文管理器的概念以及使用场景,然后介绍了with语句的作用机制,最后基于鸭子类型的理念,我们实现了两个上下文装饰器行为的魔术方法,从而实现了自定义上下文解释器的效果。
感谢您的拨冗阅读,希望对您有所帮助。
相关推荐
- oracle数据导入导出_oracle数据导入导出工具
-
关于oracle的数据导入导出,这个功能的使用场景,一般是换服务环境,把原先的oracle数据导入到另外一台oracle数据库,或者导出备份使用。只不过oracle的导入导出命令不好记忆,稍稍有点复杂...
- 继续学习Python中的while true/break语句
-
上次讲到if语句的用法,大家在微信公众号问了小编很多问题,那么小编在这几种解决一下,1.else和elif是子模块,不能单独使用2.一个if语句中可以包括很多个elif语句,但结尾只能有一个...
- python continue和break的区别_python中break语句和continue语句的区别
-
python中循环语句经常会使用continue和break,那么这2者的区别是?continue是跳出本次循环,进行下一次循环;break是跳出整个循环;例如:...
- 简单学Python——关键字6——break和continue
-
Python退出循环,有break语句和continue语句两种实现方式。break语句和continue语句的区别:break语句作用是终止循环。continue语句作用是跳出本轮循环,继续下一次循...
- 2-1,0基础学Python之 break退出循环、 continue继续循环 多重循
-
用for循环或者while循环时,如果要在循环体内直接退出循环,可以使用break语句。比如计算1至100的整数和,我们用while来实现:sum=0x=1whileTrue...
- Python 中 break 和 continue 傻傻分不清
-
大家好啊,我是大田。...
- python中的流程控制语句:continue、break 和 return使用方法
-
Python中,continue、break和return是控制流程的关键语句,用于在循环或函数中提前退出或跳过某些操作。它们的用途和区别如下:1.continue(跳过当前循环的剩余部分,进...
- L017:continue和break - 教程文案
-
continue和break在Python中,continue和break是用于控制循环(如for和while)执行流程的关键字,它们的作用如下:1.continue:跳过当前迭代,...
- 作为前端开发者,你都经历过怎样的面试?
-
已经裸辞1个月了,最近开始投简历找工作,遇到各种各样的面试,今天分享一下。其实在职的时候也做过面试官,面试官时,感觉自己问的问题很难区分候选人的能力,最好的办法就是看看候选人的github上的代码仓库...
- 面试被问 const 是否不可变?这样回答才显功底
-
作为前端开发者,我在学习ES6特性时,总被const的"善变"搞得一头雾水——为什么用const声明的数组还能push元素?为什么基本类型赋值就会报错?直到翻遍MDN文档、对着内存图反...
- 2023金九银十必看前端面试题!2w字精品!
-
导文2023金九银十必看前端面试题!金九银十黄金期来了想要跳槽的小伙伴快来看啊CSS1.请解释CSS的盒模型是什么,并描述其组成部分。...
- 前端面试总结_前端面试题整理
-
记得当时大二的时候,看到实验室的学长学姐忙于各种春招,有些收获了大厂offer,有些还在苦苦面试,其实那时候的心里还蛮忐忑的,不知道自己大三的时候会是什么样的一个水平,所以从19年的寒假放完,大二下学...
- 由浅入深,66条JavaScript面试知识点(七)
-
作者:JakeZhang转发链接:https://juejin.im/post/5ef8377f6fb9a07e693a6061目录...
- 2024前端面试真题之—VUE篇_前端面试题vue2020及答案
-
添加图片注释,不超过140字(可选)...
- 今年最常见的前端面试题,你会做几道?
-
在面试或招聘前端开发人员时,期望、现实和需求之间总是存在着巨大差距。面试其实是一个交流想法的地方,挑战人们的思考方式,并客观地分析给定的问题。可以通过面试了解人们如何做出决策,了解一个人对技术和解决问...
- 一周热门
- 最近发表
-
- oracle数据导入导出_oracle数据导入导出工具
- 继续学习Python中的while true/break语句
- python continue和break的区别_python中break语句和continue语句的区别
- 简单学Python——关键字6——break和continue
- 2-1,0基础学Python之 break退出循环、 continue继续循环 多重循
- Python 中 break 和 continue 傻傻分不清
- python中的流程控制语句:continue、break 和 return使用方法
- L017:continue和break - 教程文案
- 作为前端开发者,你都经历过怎样的面试?
- 面试被问 const 是否不可变?这样回答才显功底
- 标签列表
-
- git pull (33)
- git fetch (35)
- mysql insert (35)
- mysql distinct (37)
- concat_ws (36)
- java continue (36)
- jenkins官网 (37)
- mysql 子查询 (37)
- python元组 (33)
- mybatis 分页 (35)
- vba split (37)
- redis watch (34)
- python list sort (37)
- nvarchar2 (34)
- mysql not null (36)
- hmset (35)
- python telnet (35)
- python readlines() 方法 (36)
- munmap (35)
- docker network create (35)
- redis 集合 (37)
- python sftp (37)
- setpriority (34)
- c语言 switch (34)
- git commit (34)
