with语句与上下文管理器原理解析
编程并不是一个机械性的工作,而是需要有思考,有创新的工作,语法是固定的,但解决问题的思路则是依靠人的思维,这就需要我们坚持学习和更新自己的知识。今天golang学习网就整理分享《with语句与上下文管理器原理详解》,文章讲解的知识点主要包括,如果你对文章方面的知识点感兴趣,就不要错过golang学习网,在这可以对大家的知识积累有所帮助,助力开发能力的提升。
with语句通过上下文管理器协议确保资源在进入和退出代码块时被正确初始化和清理,即使发生异常也能自动释放资源,从而避免资源泄漏;它通过__enter__和__exit__方法或contextlib的@contextmanager装饰器实现,使文件、数据库连接等资源管理更安全、简洁。
with
语句在 Python 里,说白了,就是一种优雅地处理资源(比如文件、网络连接、锁)的方式,它能确保这些资源在使用完毕后,不管过程中有没有出错,都能被妥善地清理掉。它的魔法源自“上下文管理器”协议,这个协议定义了对象如何进入和退出一个特定的代码块。
在编程实践中,我们经常需要处理一些有“生命周期”的资源:打开文件后要关闭,连接数据库后要断开,获取锁后要释放。如果手动去管理这些步骤,尤其是在代码路径复杂或者出现异常的情况下,很容易遗漏清理操作,导致资源泄漏或者程序行为异常。with
语句就是为了解决这个痛点而生的。它通过与实现了特定方法的对象(也就是上下文管理器)协作,在进入 with
代码块时执行资源准备工作,在退出代码块时(无论正常退出还是异常退出)执行资源清理工作。这不仅让代码更简洁,也大大提高了程序的健壮性。
为什么我们要用 with
语句来处理文件或数据库连接?
我记得刚开始写Python的时候,处理文件都是这样:f = open('my_file.txt', 'r'); data = f.read(); f.close()
。后来发现,如果 f.read()
出了问题,比如文件太大内存溢出,那 f.close()
就根本执行不到,文件句柄就一直开着,时间长了,系统资源就耗尽了。或者说,如果一个函数里有多个 return
语句,你还得在每个 return
之前都加上 f.close()
,想想都头大。
这就是 with
语句的价值所在。它最核心的优势在于“异常安全”和“自动化清理”。
拿文件操作来说,当你写 with open('my_file.txt', 'r') as f:
,Python 会在文件打开后,把文件对象赋值给 f
。当 with
块内的代码执行完毕,或者在执行过程中抛出了任何异常,Python 都会自动调用文件对象的清理方法(也就是关闭文件)。你完全不用操心 f.close()
,也不用写 try...finally
这样的结构。这不仅仅是代码量减少那么简单,更重要的是,它极大地降低了因疏忽而导致资源泄漏的风险。
数据库连接也是一个典型的例子。连接池中的连接是宝贵的资源,一个应用如果不断地获取连接而不释放,很快就会把连接池耗尽,导致其他请求无法获取连接。使用 with
语句来管理数据库连接,可以确保连接在事务完成后(或发生错误时)被自动归还给连接池,或者直接关闭,避免了资源浪费。这种模式让开发者可以更专注于业务逻辑,而不是繁琐的资源管理细节。
自己动手实现一个上下文管理器需要注意什么?
实现一个自定义的上下文管理器,其实就是让你的类遵循上下文管理器协议,也就是实现两个特殊方法:__enter__
和 __exit__
。
__enter__(self)
方法会在 with
语句块被执行前调用。它的主要职责是进行资源的初始化和准备工作,并且它必须返回一个对象。这个对象就是 as
关键字后面跟着的变量所绑定的值。比如,with MyResource() as res:
,res
就会是 MyResource
实例的 __enter__
方法的返回值。如果你的资源对象本身就是你想要操作的对象,那么通常 __enter__
直接返回 self
就可以了。
__exit__(self, exc_type, exc_val, exc_tb)
方法则是在 with
语句块执行完毕后(无论是正常结束还是因为异常而中断)调用。这个方法负责资源的清理工作。它的三个参数非常关键:
exc_type
: 如果with
块内发生了异常,这里就是异常的类型(比如ValueError
)。如果没有异常,则是None
。exc_val
: 如果有异常,这里是异常实例本身。没有异常时是None
。exc_tb
: 如果有异常,这里是异常的追踪信息(traceback)。没有异常时是None
。
在 __exit__
方法里,你可以根据 exc_type
是否为 None
来判断 with
块内是否发生了异常。如果你希望在 __exit__
方法中处理掉这个异常(比如记录日志后不再向上层抛出),那么 __exit__
方法需要返回 True
。返回 True
会告诉 Python 解释器,这个异常已经被妥善处理了,不要再继续向上抛出。如果返回 False
(或者不返回任何值,因为默认就是 None
,相当于 False
),那么异常会继续传播。
举个例子,假设我们要实现一个简单的计时器上下文管理器:
import time class MyTimer: def __enter__(self): self.start_time = time.time() print("计时开始...") return self # 返回自身实例,这样 with MyTimer() as timer: 就可以用 timer 了 def __exit__(self, exc_type, exc_val, exc_tb): end_time = time.time() duration = end_time - self.start_time print(f"计时结束,总耗时:{duration:.4f} 秒") if exc_type: # 如果有异常发生 print(f"with 块内发生异常:{exc_type.__name__}: {exc_val}") # 如果我们不想让异常继续传播,可以在这里返回 True # return True # 默认返回 None,即 False,异常会继续传播 print("资源清理完成。") # 使用示例 with MyTimer(): print("正在执行一些耗时操作...") time.sleep(1.5) # raise ValueError("哦豁,出错了!") # 尝试解注释这行看看异常处理 print("with 块外部代码继续执行。")
这里 __enter__
返回了 self
,所以 with MyTimer() as timer:
这里的 timer
就是 MyTimer
的实例。__exit__
负责计算并打印耗时,同时展示了如何捕获异常信息。通过这样的设计,你可以封装任何需要“进入”和“退出”阶段的逻辑。
contextlib
模块如何简化上下文管理器的创建?
说实话,每次都写一个类,然后实现 __enter__
和 __exit__
,对于一些简单的上下文管理任务来说,确实有点小麻烦。Python 的标准库 contextlib
就是来解决这个问题的,它提供了一些工具,让创建上下文管理器变得更简单,特别是 @contextmanager
装饰器。
@contextmanager
装饰器允许你用一个生成器函数来创建上下文管理器。它的原理是,在生成器函数中,yield
关键字之前的所有代码,相当于 __enter__
方法的逻辑;yield
语句本身返回的值,就是 __enter__
方法的返回值;而 yield
语句之后的所有代码,则相当于 __exit__
方法的逻辑。
我们来用 @contextmanager
重新实现上面的计时器:
from contextlib import contextmanager import time @contextmanager def simple_timer(): start_time = time.time() print("计时开始 (通过 contextlib)...") try: yield # 这里是 with 块的代码执行的地方 except Exception as e: print(f"with 块内发生异常 (通过 contextlib):{type(e).__name__}: {e}") # 如果要抑制异常,这里可以不 re-raise,或者捕获后不抛出 # 也可以再次抛出,或者抛出新的异常 raise # 默认会重新抛出捕获到的异常 finally: end_time = time.time() duration = end_time - start_time print(f"计时结束 (通过 contextlib),总耗时:{duration:.4f} 秒") print("资源清理完成 (通过 contextlib)。") # 使用示例 with simple_timer(): print("正在执行一些耗时操作 (通过 contextlib)...") time.sleep(0.8) # raise TypeError("又出错了!") # 尝试解注释这行 print("with 块外部代码继续执行 (通过 contextlib)。")
你看,是不是简洁多了?一个函数搞定一切。yield
语句将生成器函数分割成了两个部分,完美地对应了 __enter__
和 __exit__
的行为。try...except...finally
结构在 yield
周围,使得异常处理和清理逻辑变得非常直观。如果 yield
内部的代码抛出异常,这个异常会被 try...except
捕获,你可以在 except
块中处理它。如果 except
块没有重新抛出异常,那么异常就被抑制了。finally
块则确保了清理代码无论如何都会执行。这种方式对于快速实现功能性上下文管理器,简直是神来之笔。
今天关于《with语句与上下文管理器原理解析》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于的内容请关注golang学习网公众号!

- 上一篇
- 8+32内存能混用吗?电脑内存搭配技巧

- 下一篇
- 恋与制作人第二章三星通关技巧
-
- 文章 · python教程 | 40分钟前 |
- Python图像处理:Pillow库入门教程
- 319浏览 收藏
-
- 文章 · python教程 | 1小时前 |
- Python类型提示是什么?有何优势?
- 419浏览 收藏
-
- 文章 · python教程 | 1小时前 |
- Python实现GPT-2文本生成教程
- 216浏览 收藏
-
- 文章 · python教程 | 1小时前 |
- Pandas多行更新技巧:map与update用法解析
- 168浏览 收藏
-
- 文章 · python教程 | 2小时前 |
- Python正则表达式怎么用
- 191浏览 收藏
-
- 文章 · python教程 | 3小时前 |
- Python数据库操作指南:CRUD实战解析
- 210浏览 收藏
-
- 文章 · python教程 | 3小时前 |
- Matplotlib中文乱码解决方法大全
- 466浏览 收藏
-
- 文章 · python教程 | 3小时前 |
- Python嵌套列表扁平化方法
- 332浏览 收藏
-
- 文章 · python教程 | 3小时前 |
- Python删除文件和文件夹的实用方法
- 145浏览 收藏
-
- 文章 · python教程 | 3小时前 |
- Python单引号使用技巧详解
- 283浏览 收藏
-
- 文章 · python教程 | 3小时前 |
- Python字符串转整数技巧全解析
- 140浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 514次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 499次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
-
- AI Mermaid流程图
- SEO AI Mermaid 流程图工具:基于 Mermaid 语法,AI 辅助,自然语言生成流程图,提升可视化创作效率,适用于开发者、产品经理、教育工作者。
- 615次使用
-
- 搜获客【笔记生成器】
- 搜获客笔记生成器,国内首个聚焦小红书医美垂类的AI文案工具。1500万爆款文案库,行业专属算法,助您高效创作合规、引流的医美笔记,提升运营效率,引爆小红书流量!
- 620次使用
-
- iTerms
- iTerms是一款专业的一站式法律AI工作台,提供AI合同审查、AI合同起草及AI法律问答服务。通过智能问答、深度思考与联网检索,助您高效检索法律法规与司法判例,告别传统模板,实现合同一键起草与在线编辑,大幅提升法律事务处理效率。
- 638次使用
-
- TokenPony
- TokenPony是讯盟科技旗下的AI大模型聚合API平台。通过统一接口接入DeepSeek、Kimi、Qwen等主流模型,支持1024K超长上下文,实现零配置、免部署、极速响应与高性价比的AI应用开发,助力专业用户轻松构建智能服务。
- 704次使用
-
- 迅捷AIPPT
- 迅捷AIPPT是一款高效AI智能PPT生成软件,一键智能生成精美演示文稿。内置海量专业模板、多样风格,支持自定义大纲,助您轻松制作高质量PPT,大幅节省时间。
- 601次使用
-
- Flask框架安装技巧:让你的开发更高效
- 2024-01-03 501浏览
-
- Django框架中的并发处理技巧
- 2024-01-22 501浏览
-
- 提升Python包下载速度的方法——正确配置pip的国内源
- 2024-01-17 501浏览
-
- Python与C++:哪个编程语言更适合初学者?
- 2024-03-25 501浏览
-
- 品牌建设技巧
- 2024-04-06 501浏览