上下文管理器在Python中的妙用

news2024/11/18 10:48:32

4775f2b6d7e91b4dc3893cf1d38660f2.png

更多Python学习内容:ipengtao.com

Python上下文管理器是一个非常强大的工具,它能够帮助开发者在特定代码块前后自动执行特定的操作,常用于资源管理,如文件操作、数据库连接和锁定等。本文将详细介绍Python上下文管理器的概念、使用方法、实现自定义上下文管理器,以及实际应用场景和示例代码。

什么是上下文管理器

上下文管理器是一种协议(或者说是一组方法),它允许你在进入和退出一个代码块时自动运行一些代码。上下文管理器通常与 with 语句一起使用,以确保资源在使用后被正确释放。

基本语法

with expression as variable:
    # 在这里使用变量
    pass

在这个语法中,expression 是一个返回上下文管理器的对象,variable 是上下文管理器的返回值。with 语句确保在代码块结束后,适当的清理代码会自动执行。

示例:文件操作

一个典型的使用场景是文件操作。使用上下文管理器可以确保文件在使用后被正确关闭。

# 不使用上下文管理器
file = open('example.txt', 'r')
try:
    content = file.read()
finally:
    file.close()

# 使用上下文管理器
with open('example.txt', 'r') as file:
    content = file.read()

在上面的示例中,with 语句确保文件在读取完毕后自动关闭,即使在读取过程中发生异常。

内置上下文管理器

Python内置了多个上下文管理器,常见的包括文件操作、线程锁和数据库连接等。

示例:使用锁

import threading

lock = threading.Lock()

# 不使用上下文管理器
lock.acquire()
try:
    # 访问共享资源
    pass
finally:
    lock.release()

# 使用上下文管理器
with lock:
    # 访问共享资源
    pass

在这个示例中,with 语句确保锁在访问共享资源后自动释放。

实现自定义上下文管理器

可以通过实现 __enter____exit__ 方法来自定义上下文管理器。这两个方法分别在进入和退出上下文时被调用。

示例:自定义上下文管理器

class MyContextManager:
    def __enter__(self):
        print("进入上下文")
        return self

    def __exit__(self, exc_type, exc_value, traceback):
        print("退出上下文")

# 使用自定义上下文管理器
with MyContextManager() as manager:
    print("在上下文中")

在这个示例中,__enter__ 方法在进入上下文时打印消息,__exit__ 方法在退出上下文时打印消息。

使用 contextlib 模块

Python的 contextlib 模块提供了更方便的方式来创建上下文管理器,特别是对于简单的情况。

示例:使用 contextlib.contextmanager 装饰器

from contextlib import contextmanager

@contextmanager
def my_context_manager():
    print("进入上下文")
    yield
    print("退出上下文")

# 使用自定义上下文管理器
with my_context_manager():
    print("在上下文中")

在这个示例中,@contextmanager 装饰器将一个生成器函数转换为上下文管理器。yield 语句将控制权传递给 with 语句内部的代码块,代码块执行完毕后,执行 yield 之后的代码。

上下文管理器的实际应用

示例:计时器上下文管理器

一个常见的应用场景是创建一个计时器上下文管理器,用于测量代码块的执行时间。

import time
from contextlib import contextmanager

@contextmanager
def timer():
    start_time = time.time()
    yield
    end_time = time.time()
    print(f"执行时间: {end_time - start_time} 秒")

# 使用计时器上下文管理器
with timer():
    # 模拟长时间运行的任务
    time.sleep(2)

在这个示例中,计时器上下文管理器测量代码块的执行时间,并在退出上下文时打印执行时间。

示例:数据库连接上下文管理器

另一个实际应用场景是管理数据库连接,确保在操作数据库后正确关闭连接。

import sqlite3
from contextlib import contextmanager

@contextmanager
def db_connection(db_name):
    conn = sqlite3.connect(db_name)
    try:
        yield conn
    finally:
        conn.close()

# 使用数据库连接上下文管理器
with db_connection('example.db') as conn:
    cursor = conn.cursor()
    cursor.execute('SELECT * FROM example_table')
    rows = cursor.fetchall()
    for row in rows:
        print(row)

在这个示例中,数据库连接上下文管理器确保在操作数据库后正确关闭连接。

处理异常

上下文管理器还可以用于处理异常,确保在发生异常时执行必要的清理操作。

示例:处理异常的上下文管理器

class ExceptionHandlingContextManager:
    def __enter__(self):
        print("进入上下文")
        return self

    def __exit__(self, exc_type, exc_value, traceback):
        if exc_type:
            print(f"异常类型: {exc_type}")
            print(f"异常值: {exc_value}")
            print("处理异常")
            return True  # 表示异常已经被处理
        print("退出上下文")
        return False  # 表示异常未被处理

# 使用异常处理上下文管理器
with ExceptionHandlingContextManager() as manager:
    print("在上下文中")
    raise ValueError("这是一个示例异常")

在这个示例中,__exit__ 方法处理了在上下文中发生的异常,并返回 True 表示异常已经被处理。

总结

本文详细介绍了Python上下文管理器的概念、使用方法、自定义实现以及实际应用场景。通过内置和自定义上下文管理器,开发者可以在特定代码块前后自动执行特定操作,如资源管理和异常处理,确保代码更加优雅和高效。文中提供了丰富的示例代码,展示了文件操作、线程锁、计时器和数据库连接等常见应用。掌握上下文管理器的使用,可以显著提高代码的可读性和可维护性。

如果你觉得文章还不错,请大家 点赞、分享、留言 下,因为这将是我持续输出更多优质文章的最强动力!


如果想要系统学习Python、Python问题咨询,或者考虑做一些工作以外的副业,都可以扫描二维码添加微信,围观朋友圈一起交流学习。

3abaa3a5689ba55c974da37f38e132b1.gif

我们还为大家准备了Python资料和副业项目合集,感兴趣的小伙伴快来找我领取一起交流学习哦!

c750173fe3d99ded363bb8a02c3caa8c.jpeg

往期推荐

历时一个月整理的 Python 爬虫学习手册全集PDF(免费开放下载)

Python基础学习常见的100个问题.pdf(附答案)

学习 数据结构与算法,这是我见过最友好的教程!(PDF免费下载)

Python办公自动化完全指南(免费PDF)

Python Web 开发常见的100个问题.PDF

肝了一周,整理了Python 从0到1学习路线(附思维导图和PDF下载)

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

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

相关文章

Arduino - MG996R

Arduino - MG996R In this tutorial, we are going to learn how to use the MG996R high-torque servo motor with Arduino. 在本教程中,我们将学习如何将MG996R高扭矩伺服电机与Arduino一起使用。 Hardware Required 所需硬件 1Arduino UNO or Genuino UNO Ard…

Python终于可以在线编程了!

优势 在线编程,轻量级,无需安装Python环境。 在线编程优势: 无需安装和配置环境: 在线编程平台不需要用户在本地安装任何软件或配置开发环境。这对初学者和那些希望快速上手进行编程的人非常有利。跨平台兼容性: 这些平台可以在任何具有互联网…

java将html转成图片

java 将html转成图片 1.导入jar2.代码3.展示结果4.注意事项 最近有一个需求需要根据指定的样式生成图片&#xff0c;使用java原生技术有些麻烦&#xff0c;所以上网搜了下案例&#xff0c;最后发现最好用的还是html2image&#xff0c;这里进行简单总结下。 1.导入jar <!-- 用…

软件开发方法

软件开发方法 瀑布方法优势 敏捷法优势敏捷软件开发原则 激进&#xff08;Scrum&#xff09;优势 极限编程优势 精益优势 看板优势 迭代增量模型 创建软件并不是一件简单的事情&#xff1a;通常&#xff0c;开发应用程序需要不同技能的团队协同努力。如果没有战略管理&#xff…

一个去掉PDF背景水印的思路

起因 昨天测试 使用“https://github.com/VikParuchuri/marker” 将 pdf 转 Markdown的过程中&#xff0c;发现转换后的文件中会保护一些背景图片&#xff0c;是转换过程中&#xff0c;程序把背景图识别为了内容。于是想着怎么把背景图片去掉。 背景水印图片的特征 我这里拿…

2024软件设计师笔记之考点版(一考就过):26-39

软件设计师之一考就过:成绩版 考点26:类、封装、继承、多态 真题1:在面向对象方法中,两个及以上的类作为一个类的超类时,称为(多重继承),使用它可能造成子类中存在(二义性)的成员。 真题2:在面向对象方法中,多态指的是(客户类无需知道所调用方法的特定子类的实现…

SwiftUI 6.0(iOS/iPadOS 18)中全新的 Tab 以及 Sidebar+悬浮 TabView 样式

概览 看来苹果一直对 iPadOS 中标签栏&#xff08;TabView&#xff09;不甚满意。这不&#xff0c;在 WWDC 2024 中苹果又对 TabView 外观做了大幅度的进化。 现在我们可以在顶部悬浮条和左侧的 Sidebar 两种不同布局之间恣意切换 TabView 的外观啦。而且&#xff0c;这在 Swi…

ubuntu 18 虚拟机安装(3)安装mysql

ubuntu 18 虚拟机安装&#xff08;3&#xff09;安装mysql 参考 https://cloud.tencent.com/developer/article/1700780 技术分享 | MySQL 设置管理员密码无法生效一例 https://cloud.tencent.com/developer/article/2014384 在Ubuntu18.04上安装MySQL &#xff5c; 超级详细…

字节豆包 MarsCode:AI 开发工具

MarsCode 是豆包旗下的智能编程助手&#xff0c;类似 GitHub Copilot 提供以智能代码补全为代表的核心能力&#xff0c;简单试用了下&#xff0c;免费&#xff0c;使用时需要手机号登录&#xff0c;代码补全还算 ok&#xff0c;聊天功能就有点差了。 还包括一个 AI 原生 IDE&am…

EPLAN批量修改文字大小

在项目设计过程中&#xff0c;往往要批量调整文字的大小&#xff0c;如何批量修改文字大小&#xff1a; 点击需要调整的相同类的文字 右键 “属性”&#xff0c;然后在分配里找到“属性放置&#xff0c;设备标识符”这一栏 看下文字的属性在第几层 在项目数据找到层管理&…

道路救援入驻派单小程序开源版开发

道路救援入驻派单小程序开源版开发 1、用户立即救援 2、后台收到救援通知&#xff0c;派单救援师傅. 道路救援入驻派单小程序通常会包含一系列功能&#xff0c;旨在方便救援服务提供商、用户和后台管理系统之间的交互。以下是一个可能的功能列表&#xff1a; 用户端功能&…

前端框架中的前端打包(Bundling)和前端构建工具(Build Tools)的作用

聚沙成塔每天进步一点点 本文回顾 ⭐ 专栏简介前端框架中的前端打包&#xff08;Bundling&#xff09;和前端构建工具&#xff08;Build Tools&#xff09;的作用1. 引言2. 前端打包&#xff08;Bundling&#xff09;2.1 概述2.2 常见的打包工具2.2.1 Webpack2.2.2 Parcel 2.3 …

合约期VS优惠期,搞明白他们的区别才能避免很多坑!

在购买流量卡时&#xff0c;相信大家也都发现了&#xff0c;市面上的不少套餐都是有合约期和优惠期的&#xff0c;尤其是联通和移动&#xff0c;那么&#xff0c;什么是合约期&#xff1f;什么又是优惠期呢&#xff1f; ​ 其实&#xff0c;目前很多在网上办理的大流量卡都是有…

静态图和动态图中的自动求导机制详解

01 静态图与动态图的区别 之前在 [1] 中提到过&#xff0c;自动求导&#xff08;AutoDiff&#xff09;机制是当前深度学习模型训练采用的主要方法&#xff0c;而在静态图和动态图中对于自动求导的处理是不一样的。作为前置知识&#xff0c;这里简单进行介绍。 我们都知道静态…

【深度学习】tensorboard的使用

目前正在写一个训练框架&#xff0c;需要有以下几个功能&#xff1a; 1.保存模型 2.断点继续训练 3.加载模型 4.tensorboard 查询训练记录的功能 命令&#xff1a; tensorboard --logdirruns --host192.168.112.5 效果&#xff1a; import torch import torch.nn as nn impor…

排序算法(2)之选择排序----直接选择排序和堆排序

个人主页&#xff1a;C忠实粉丝 欢迎 点赞&#x1f44d; 收藏✨ 留言✉ 加关注&#x1f493;本文由 C忠实粉丝 原创 排序算法(2)之交换排序----冒泡排序和堆排序 收录于专栏【数据结构初阶】 本专栏旨在分享学习数据结构学习的一点学习笔记&#xff0c;欢迎大家在评论区交流讨论…

【系统架构设计师】七、信息安全技术基础知识(信息安全的概念|信息安全系统的组成框架|信息加解密技术)

目录 一、信息安全的概念 1.1 信息安全的基本要素和范围 1.2 信息存储安全 1.3 网络安全 二、信息安全系统的组成框架 2.1 技术体系 2.2 组织机构体系 2.3 管理体系 三、 信息加解密技术 3.1 数据加密 3.2 对称加密技术 3.3 非对称加密算法 3.4 数字信封 3.5 信…

信息系统项目管理师(项目管理师)

项目管理者再坚持“聚焦于价值”原则时&#xff0c;应该关注的关键点包括&#xff1a;1价值是项目成功的最终指标&#xff1b;2价值可以再整个项目进行期间、项目结束或完成后实现&#xff1b;3价值可以从定性和/或定量的角度进行定义和衡量&#xff1b;4以成果为导向&#xff…

鸿蒙开发系统基础能力:【@ohos.pasteboard (剪贴板)】

剪贴板 说明&#xff1a; 本模块首批接口从API version 6开始支持。后续版本的新增接口&#xff0c;采用上角标单独标记接口的起始版本。 导入模块 import pasteboard from ohos.pasteboard;属性 系统能力: 以下各项对应的系统能力均为SystemCapability.MiscServices.Pasteb…

ubuntu 18 虚拟机安装(1)

ubuntu 18 虚拟机安装 ubuntu 18.04.6 Ubuntu 18.04.6 LTS (Bionic Beaver) https://releases.ubuntu.com/bionic/ 参考&#xff1a; 设置固定IP地址 https://blog.csdn.net/wowocpp/article/details/126160428 https://www.jianshu.com/p/1d133c0dec9d ubuntu-18.04.6-l…