Python中with的作用和用法

在这里我们来详细解释一下Python中非常重要的 with 语句。

我会从 “为什么需要它” 开始,然后讲解 “它是什么以及如何使用”,最后深入到 “它的工作原理”“如何自定义”


1. 为什么需要 with 语句?(The Problem)

在编程中,我们经常会使用一些需要“获取”和“释放”的资源,比如:

  • 文件操作:打开文件后,必须记得关闭它。
  • 数据库连接:建立连接后,必须记得关闭连接。
  • 线程锁:获取锁之后,必须记得释放它。

如果我们忘记释放这些资源,可能会导致严重的问题,比如:

  • 文件句柄耗尽,无法再打开新文件。
  • 数据库连接池被占满,应用无法再连接数据库。
  • 线程死锁,程序卡住。

让我们看一个没有 with 的文件操作例子:

不安全的写法:

f = open('my_file.txt', 'w')
f.write('hello world')
# 如果在 write 和 close 之间发生错误,close() 将永远不会被执行!
f.close()

这个写法非常危险。如果在 f.write() 时发生异常(例如磁盘满了),程序会崩溃,f.close() 就不会被调用,文件资源就泄露了。

安全的、但繁琐的写法 (使用 try...finally):
为了确保资源一定被释放,我们通常使用 try...finally 结构:

f = None # 在 try 外面初始化,确保 finally 中可以访问
try:f = open('my_file.txt', 'w')f.write('hello world')# ... 其他可能出错的操作 ...
finally:if f:f.close()

这个写法是安全的,因为无论 try 块中是否发生异常,finally 块中的代码都保证会被执行。但是,它看起来很冗长,代码结构也不够优雅。

with 语句就是为了解决这个问题而生的,它能让我们用更简洁、更安全的方式来管理资源。


2. with 语句是什么以及如何使用?(The Solution)

with 语句是一种上下文管理的语法糖(Syntactic Sugar)。它极大地简化了上面 try...finally 的写法。

基本语法:

with expression as variable:# 在这个代码块中,资源是可用的# ... do something with variable ...# 离开 with 代码块后,资源会自动被清理

使用 with 重写文件操作:

with open('my_file.txt', 'w') as f:f.write('hello world')# 在这里可以进行各种文件操作# 比如 f.read(), f.writelines() 等# 当代码执行离开这个 with 块时(无论是正常结束还是发生异常),
# Python 会自动调用 f.close(),我们完全不需要操心。

对比一下:

  • try...finally 版本:5-6 行代码,结构复杂。
  • with 版本:2 行代码,逻辑清晰,意图明确(“在处理这个文件的上下文中,做这些事”)。

with 语句的核心优势是:无论 with 块内部发生什么(即使是异常),它都保证能执行资源的“清理”操作


3. with 的工作原理:上下文管理器协议 (The Magic Behind)

with 语句之所以能自动管理资源,是因为它遵循了上下文管理器协议(Context Manager Protocol)

一个对象只要实现了下面这两个特殊方法,它就是一个上下文管理器:

  1. __enter__(self)

    • 何时调用:当进入 with 语句块时,该方法被调用。
    • 作用:负责“获取”资源或进行初始化设置。
    • 返回值:这个方法的返回值会赋给 as 后面的变量(如果 as 存在的话)。如果你不需要 as 变量,这个方法可以不返回任何东西。
  2. __exit__(self, exc_type, exc_value, traceback)

    • 何时调用:当离开 with 语句块时(无论是正常退出还是因为异常退出),该方法被调用。
    • 作用:负责“释放”资源或执行清理操作(比如 f.close())。
    • 参数
      • exc_type: 异常的类型(如果没发生异常,则为 None)。
      • exc_value: 异常的值(如果没发生异常,则为 None)。
      • traceback: 异常的追溯信息(如果没发生异常,则为 None)。
    • 返回值
      • 如果 __exit__ 方法返回 True,表示它已经处理了这个异常,异常会被“吞掉”(suppress),程序不会向外抛出。
      • 如果它返回 FalseNone(默认情况),任何发生的异常都会在 __exit__ 执行完毕后被重新抛出。

所以,with open(...) as f: 这段代码大致等同于下面的伪代码:

# 1. 创建上下文管理器对象
manager = open('my_file.txt', 'w')# 2. 调用 __enter__ 方法,返回值赋给 f
f = manager.__enter__()# 3. 执行 with 块中的代码
try:f.write('hello world')
finally:# 4. 无论如何,都调用 __exit__ 方法进行清理# (这里简单展示,实际会传递异常信息)manager.__exit__(None, None, None)

4. 如何创建自己的上下文管理器?

了解了原理,我们就可以创建自己的上下文管理器。有两种主要方式:

方式一:基于类的实现

我们可以写一个类,并实现 __enter____exit__ 方法。

示例:一个简单的计时器

import timeclass Timer:def __init__(self, name):self.name = namedef __enter__(self):print(f"计时器 '{self.name}' 开始...")self.start_time = time.time()# 这个类本身就是资源,所以返回 selfreturn self def __exit__(self, exc_type, exc_value, traceback):self.end_time = time.time()duration = self.end_time - self.start_timeprint(f"计时器 '{self.name}' 结束,耗时: {duration:.4f} 秒")# 如果有异常,这里可以记录日志if exc_type:print(f"在 '{self.name}' 中发生了异常: {exc_value}")# 返回 False 或 None,让异常正常抛出return False# 使用自定义的 Timer
with Timer("数据处理") as t:print("正在处理数据...")time.sleep(2)print("数据处理完成。")print("-" * 20)with Timer("有问题的操作") as t:print("准备执行一个会出错的操作...")time.sleep(1)result = 1 / 0  # 这里会产生一个 ZeroDivisionErrorprint("这行代码不会被执行")

输出:

计时器 '数据处理' 开始...
正在处理数据...
数据处理完成。
计时器 '数据处理' 结束,耗时: 2.0021 秒
--------------------
计时器 '有问题的操作' 开始...
准备执行一个会出错的操作...
计时器 '有问题的操作' 结束,耗时: 1.0011 秒
在 '有问题的操作' 中发生了异常: division by zero
Traceback (most recent call last):File "...", line 36, in <module>result = 1 / 0  # 这里会产生一个 ZeroDivisionError
ZeroDivisionError: division by zero

可以看到,即使发生了异常,__exit__ 方法仍然被调用,成功打印了耗时和异常信息。

方式二:基于生成器的实现(使用 contextlib 模块)

对于简单的上下文管理器,每次都写一个类有点麻烦。Python 的 contextlib 模块提供了一个 @contextmanager 装饰器,可以让我们用更简洁的方式实现。

import time
from contextlib import contextmanager@contextmanager
def timer(name):print(f"计时器 '{name}' 开始...")start_time = time.time()# yield 之前的部分,相当于 __enter__# yield 的值会成为 as 后面的变量(如果没有 yield 值,则为 None)try:yieldfinally:# yield 之后的部分,相当于 __exit__end_time = time.time()duration = end_time - start_timeprint(f"计时器 '{name}' 结束,耗时: {duration:.4f} 秒")# 使用方法完全一样
with timer("数据处理_v2"):print("正在处理数据...")time.sleep(2)print("数据处理完成。")

这种方式更加 Pythonic,代码也更紧凑。try...yield...finally 结构完美地对应了“进入-执行-清理”的模式。


总结

  • 用途with 语句用于自动管理资源,确保资源在使用完毕后(无论是否发生异常)都能被正确清理。
  • 优点:代码更简洁、更安全、更具可读性,避免了冗长的 try...finally 结构和资源泄露的风险。
  • 原理:依赖于上下文管理器协议,即对象需实现 __enter__()__exit__() 两个方法。
  • 自定义:你可以通过编写类或使用 contextlib.contextmanager 装饰器来创建自己的上下文管理器,封装任何需要“设置-清理”逻辑的场景。

在现代 Python 编程中,只要遇到需要获取和释放资源的场景,都应该优先考虑使用 with 语句。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/bicheng/89430.shtml
繁体地址,请注明出处:http://hk.pswp.cn/bicheng/89430.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

缓存雪崩、缓存穿透,缓存击穿

Redis是一个完全开源免费的高性能非关系型&#xff08;NOSQL&#xff09;的key-value数据库。 Redis不可能把所有的数据都缓存起来(内存昂贵且有限)&#xff0c;所以Redis需要对数据 设置过期时间&#xff0c;并采用的是惰性删除定期删除两种策略对过期键删除。Redis对过期键的…

springmvc跨域解决方案

在Spring MVC中处理跨域请求&#xff08;CORS&#xff0c;Cross-Origin Resource Sharing&#xff09;通常涉及到配置HTTP响应头&#xff0c;以允许来自不同源的请求。Spring MVC提供了多种方式来配置CORS&#xff0c;包括全局配置和局部配置。 使用CrossOrigin注解 在控制器的…

btstack移植之安全配对(二)

3.13.3 Legacy配对首先&#xff0c;我们回复的paring response中&#xff0c;可以看到我们不支持secure connection&#xff0c;所以我们走的是legacy配对模式。图3-74 secure连接不支持然后&#xff0c;master在pairing confirm包中回复了confirm value。图3-75 master发送con…

FRP配置( CentOS 7 上安装 FRP教程 )

** 如果你们公司没有公网IP&#xff0c;但是又想实现内网穿透&#xff0c;远程调用接口&#xff0c;在家也能调用公司服务器&#xff0c;但是nkg ssl有问题&#xff0c;花生壳坑壁&#xff0c;那么FRP是你最佳的选择&#xff01;&#xff01;&#xff01;** 不过有个前提&#…

第三次mysql作业

建立库建立mydb11_syu库2.创建s表&#xff0c;创建sc表二&#xff0e;插入数据向s表插入数据2.向sc表插入数据三&#xff0e;查询1.分别查询student表和score表的所有记录2.查询student表的第2条到5条记录3.从student表中查询计算机系和英语系的学生的信息4.从student表中查询年…

不同场景下git指令的搭配

添加账号 git config --global user.name "YourName" git config --global user.email "your_emailexample.com"设置 Git 默认分支名称为 main&#xff1a; git config --global init.defaultBranch main初始化仓库&#xff1a; git init配置SSH 密钥 如果…

NLP——迁移学习

一、迁移学习的概念 1.预训练模型(Pretrained model) 定义: 简单来说别人训练好的模型。一般预训练模型具备复杂的网络模型结构&#xff1b;一般是在大量的语料下训练完成的。 2.微调(Fine-tuning) 定义:一般是对预训练语言模型&#xff0c;进行垂直领域数据的微调&#xff0c;…

Ubuntu 安装

文章目录硬件准备下载 Ubuntu 镜像创建可启动 USB从 USB 驱动器启动安装 Ubuntu不要忘记更新&#xff01;用了十多年的笔记本&#xff0c;手边正好有个500G的固态&#xff0c;准备换上。考虑到机器的硬件配置&#xff0c;现在使用windows10实在是有点卡&#xff0c;ubuntu却刚好…

【46】MFC入门到精通——MFC显示实时时间,获取系统当前时间GetCurrentTime()、获取本地时间GetLocalTime()

文章目录1 MFC获取时间方法方法一&#xff1a;获取系统当前时间GetCurrentTime()方法二&#xff1a;获取本地时间GetLocalTime()使用GetTickCount()获取程序运行时间2 MFC显示实时时间 使用方法2.1 获取时间2.2 类向导 添加定时器函数 OnTimer2.3 初始化 设置定时器2.4 定时器函…

Linux717 SWAP扩容;逻辑卷条带化

root192.168.235.20s password:┌────────────────────────────────────────────────────────────────────┐│ • MobaXterm 20.0 • ││ …

人类社会发展过程中的熵增定律

引子研究美国罗斯福新政期间的法案为什么会对美国经济恢复起作用&#xff1f;与千金买马骨和移木赏金之间的区别与联系&#xff1f;以下为豆包 AI回答一、罗斯福新政法案对美国经济起作用的原因罗斯福新政&#xff08;1933-1939年&#xff09;通过一系列政策应对大萧条&#x…

Spring-AI系列-入门篇-核心概念、组件和生产要素

原文-Spring AI知识库&#xff0c;欢迎大家评论互动 师父领进门&#xff0c;修行靠自己。 Spring AI is an application framework for AI engineering. Its goal is to apply to the AI domain Spring ecosystem design principles such as portability and modular design an…

从浏览器到服务器:TCP 段的网络传输之旅

本文以简化的网络架构为例&#xff0c;详细介绍了当你在浏览器中输入网址&#xff08;例如www.google.com&#xff09;并按下回车键后&#xff0c;TCP段的完整传输过程。我们将探讨DNS解析、ARP、TCP/IP封装、PAT和路由如何协同工作&#xff0c;将数据从个人电脑通过局域网和广…

HCIE - 云计算拿下后的职业选择如何规划?

Hello&#xff01;大家好&#xff0c;小编是一名专注 IT 领域的资深探索家。我们聊聊HCIE - 云计算&#xff0c;这个认证作为华为认证体系中云计算领域的专家级认证&#xff0c;标志着持有者具备企业级云架构设计、复杂云平台运维及跨场景技术落地能力。但认证本身只是职业进阶…

1-创建Vue3项目

创建Vue3项目前提 已安装 18.3 或更高版本的 Node.js vue 官网 https://vuejs.org/ 创建一个 Vue 应用 ① 新建项目目录&#xff0c;使用 VSCode 打开 VSCode 可安装 Vue-Official 插件协助开发 ② 执行 create vue 指令创建 vue 应用 npm create vuelatest这一指令将…

Codex,Copilot 是什么

Codex是什么 Codex 是 OpenAI 研发的一款专注于代码生成的大型语言模型,它可以根据自然语言描述自动编写程序代码,在软件开发、自动化测试等领域展现出了强大的应用潜力。下面为你详细介绍: 1. 核心功能 代码生成:Codex 能够依据自然语言指令生成代码,像函数、类或者完整…

Typecho插件开发:自定义表单验证规则addRule实战指南

文章目录 Typecho表单验证进阶:为插件和主题添加自定义addRule验证规则 引言 一、Typecho表单验证基础 1.1 Typecho表单系统概述 1.2 addRule方法解析 二、自定义验证规则实现 2.1 创建自定义验证类 2.2 注册自定义验证规则 2.3 使用自定义验证规则 三、高级验证场景实现 3.1 …

数据分布是如何影响目标检测精度的

文章目录一、研究背景与目标模型效果提升数据集优化二、研究问题明细各方向的关联性与核心逻辑1. 高质量数据集的高效筛选与主动学习应用2. 基于推理结果的数据补充与增强方向优化3. 多类别场景下目标尺度与模型精度的关联性4. 损失函数与数据增强对精度的量化影响5. 目标类型专…

Python 网络爬虫 —— 代理服务器

一、会话&#xff08;Session&#xff09;&#xff08;一&#xff09;核心逻辑HTTP 本身无记忆&#xff0c;每次请求独立。会话&#xff08;Session&#xff09; 就是为解决这问题&#xff0c;让客户端&#xff08;浏览器&#xff09;和服务器 “记住” 交互状态&#xff08;比…

Vue在线预览Excel和Docx格式文件

前提&#xff1a;本次示例基于Vue2.x&#xff0c;所用插件为Vue-Office。 一、Vue-Office 插件简介 Vue-Office 是一个一站式解决方案&#xff0c;支持多种 Office 文件格式的在线预览&#xff0c;包括&#xff1a; Word&#xff08;.docx&#xff09;Excel&#xff08;.xlsx、…