python with使用详解

 更新时间:2025年11月03日 10:07:06   作者:令狐掌门  
with语句是Python中一种用于资源管理的语法结构,主要用于确保资源(如文件、锁、数据库连接等)在使用后被正确释放,即使发生异常也不会遗漏清理操作,本文给大家介绍python with使用,感兴趣的朋友跟随小编一起看看吧

1. Pythonwith语句的原理

with语句是Python中一种用于资源管理的语法结构,主要用于确保资源(如文件、锁、数据库连接等)在使用后被正确释放,即使发生异常也不会遗漏清理操作。它本质上是基于上下文管理器(Context Manager)协议实现的。

核心原理:

  • with语句会调用上下文管理器的两个特殊方法:
    • __enter__():进入with块时调用,返回一个值(通常是资源对象本身),这个值可以被as关键字赋值给变量。
    • __exit__(exc_type, exc_value, traceback):退出with块时调用,无论是否发生异常都会执行。它接收三个参数(异常类型、异常值、回溯信息),如果返回True,则抑制异常(不抛出);否则,异常会继续传播。
  • with语句相当于简化版的try...finally结构:
    • 进入时执行__enter__()(类似于try前的准备)。
    • 执行with块内的代码。
    • 退出时执行__exit__()(类似于finally块中的清理)。
  • 好处:自动处理资源释放,避免手动编写try...except...finally块,减少错误(如忘记关闭文件)。

如果没有异常,__exit__的参数均为None。这使得with特别适合“获取-使用-释放”模式的操作。

2.with语句的用法

with语句的基本语法:

with 上下文管理器 [as 变量]:
    # 在这里使用资源
    # 块结束时自动清理

示例1:文件操作(最常见用法)

with open('example.txt', 'r') as file:
    content = file.read()
    print(content)
# 文件在with块结束后自动关闭,无需手动调用file.close()
  • 原理:open()返回的文件对象实现了上下文管理器协议,__enter__返回文件对象本身,__exit__关闭文件。

示例2:多上下文管理器

Python支持嵌套或并列使用多个with

with open('input.txt', 'r') as infile, open('output.txt', 'w') as outfile:
    content = infile.read()
    outfile.write(content.upper())

示例3:处理异常

try:
    with open('nonexistent.txt', 'r') as file:
        print(file.read())
except FileNotFoundError:
    print("文件不存在")
  • 如果with块内抛出异常,__exit__仍会执行清理,然后异常传播到外层。

with常用于:

  • 文件I/O(open())。
  • 线程锁(threading.Lock())。
  • 数据库连接(许多库如SQLAlchemy支持)。
  • 临时更改(如os.chdir()的上下文版本)。

3. 如何定义一个对象使其支持with语句

要让自定义对象支持with,需要实现上下文管理器协议。有两种主要方式:类实现生成器实现(使用contextlib模块)。

方式1:通过类实现__enter__和__exit__

定义一个类,添加__enter____exit__方法。

class MyResource:
    def __init__(self, name):
        self.name = name
        print(f"初始化资源: {name}")
    def __enter__(self):
        print(f"进入with块: 获取资源 {self.name}")
        return self  # 返回对象本身,供as使用
    def __exit__(self, exc_type, exc_value, traceback):
        print(f"退出with块: 释放资源 {self.name}")
        if exc_type is not None:
            print(f"发生异常: {exc_type}")
            return False  # 不抑制异常,让它传播
        return True  # 正常退出
# 用法
with MyResource("数据库连接") as res:
    print(f"使用资源: {res.name}")
    # raise ValueError("模拟异常")  # 可以测试异常处理

输出示例(无异常):

初始化资源: 数据库连接
进入with块: 获取资源 数据库连接
使用资源: 数据库连接
退出with块: 释放资源 数据库连接

方式2:使用contextlib模块的@contextmanager装饰器(基于生成器)

这是一种更简洁的方式,适合临时上下文,不需要定义整个类。

from contextlib import contextmanager
@contextmanager
def my_resource(name):
    print(f"进入with块: 获取资源 {name}")
    yield name.upper()  # yield前的代码相当于__enter__,yield的值供as使用
    print(f"退出with块: 释放资源 {name}")  # yield后的代码相当于__exit__
# 用法
with my_resource("文件锁") as res:
    print(f"使用资源: {res}")

输出:

进入with块: 获取资源 文件锁
使用资源: 文件锁
退出with块: 释放资源 文件锁

  • 处理异常:在生成器中,可以用try...except包裹yield,以自定义异常行为。

注意事项:

  • __enter__返回的值可以是任何对象,不一定是self(例如,返回一个计算结果)。
  • __exit__中,如果要抑制异常,返回True;否则返回FalseNone
  • 如果对象不支持上下文管理器,使用with会抛出AttributeError(缺少__enter____exit__)。
  • Python 3.10+ 支持异步上下文管理器(__aenter____aexit__),用于async with

通过这些方式,可以为任何需要资源管理的场景自定义with支持。

到此这篇关于python with使用介绍的文章就介绍到这了,更多相关python with使用内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 详解Django CAS 解决方案

    详解Django CAS 解决方案

    这篇文章主要介绍了详解Django CAS 解决方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-10-10
  • Python使用pymysql模块操作mysql增删改查实例分析

    Python使用pymysql模块操作mysql增删改查实例分析

    这篇文章主要介绍了Python使用pymsql模块操作mysql增删改查,结合实例形式分析了Python使用pymsql模块针对mysql进行增删改查操作的相关实现方法与操作注意事项,需要的朋友可以参考下
    2019-12-12
  • 用python代码将tiff图片存储到jpg的方法

    用python代码将tiff图片存储到jpg的方法

    今天小编就为大家分享一篇用python代码将tiff图片存储到jpg的方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-12-12
  • vscode带命令行参数进行调试的方法

    vscode带命令行参数进行调试的方法

    文章介绍了如何在VSCode中使用命令行参数进行调试,并描述了如何通过修改`launch.json`文件来简化调试过程
    2025-01-01
  • Django的用户模块与权限系统的示例代码

    Django的用户模块与权限系统的示例代码

    这篇文章主要介绍了Django的用户模块与权限系统的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-07-07
  • Python代码轻松搞定PDF页面方向调整的详细教学

    Python代码轻松搞定PDF页面方向调整的详细教学

    Python作为自动化办公的利器,能通过几行代码批量解决PDF页面方向调整问题,本文将用通俗易懂的方式带你掌握如何精准控制页面方向,感兴趣的小伙伴可以了解下
    2026-02-02
  • python爬取内容存入Excel实例

    python爬取内容存入Excel实例

    这篇文章主要为大家详细介绍了python爬取内容存入Excel实例,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-02-02
  • python requests指定出口ip的例子

    python requests指定出口ip的例子

    今天小编就为大家分享一篇python requests指定出口ip的例子,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2019-07-07
  • 聊聊python中not 与 is None的区别

    聊聊python中not 与 is None的区别

    这篇文章主要介绍了在python中not 与 is None的区别,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-05-05
  • Python3 中作为一等对象的函数解析

    Python3 中作为一等对象的函数解析

    这篇文章主要介绍了Python3 中作为一等对象的函数,本文通过实例代码讲解的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下
    2019-12-12

最新评论