Python上下文管理器详解

简介: Python上下文管理器详解

Python上下文管理器详解

首先,看一个简单的例子:

# 创建一个文件写入字符串“Python”
f = open('123.txt', 'w')
f.write("python")
f.close()


# 使用with语句调用上下文实现文件写入操作
with open('123.txt', 'w') as f:
    f.write('python')

可以看到,使用上下文管理器,可以让代码更加优雅简洁。当然,上下文的管理器的作用不止于此,它内部的实现机制,能很好的处理代码异常,提升代码的复用性。

什么是上下文管理器?

这里先了解下上下文,上下文是 context 直译的叫法,在程序中用来表示代码执行过程中所处的前后环境,比如在文件操作时,文件需要打开关闭,而文件读写操作就处于文件操作的上下文环境中。

上下文管理器,上下文管理器是指在一段代码执行之前,执行一些预处理的工作,代码执行之后再执行一些清理工作。

上下文管理器中有__enter__()和 __exit__() 两个方法,__enter__()方法在执行 with 后面的语句时执行,一般用来处理操作前的内容,比如一些创建对象,初始化等; __exit__() 方法在 with 内的代码执行完毕后执行,一般用来处理一些善后收尾工作,比如文件的关闭,数据库的关闭等。

简单来说,上下文管理器的原理过程如下:

  • 调用__enter__()方法,进行预处理操作
  • 执行用户操作
  • 调用 __exit__() 方法,完成清理操作

上下文管理器的应用场景:

资源管理功能,即文件处理、网络连接、数据库连接等操作时需要关闭资源。

也可以在代码执行前后增加功能,类似于装饰器,比如代码之前做权限验证等。

一个数据库连接的使用:

import pymysql
class DBConnection(object):
    def __init__(self,ip,user,passwd,db):
        self.ip = ip
        self.user = user
        self.passwd = passwd
        self.db = db

    def __enter__(self):
        self.conn = pymysql.connect(self.ip, user=self.user, passwd=self.passwd, db=self.db)
        self.cur = conn.cursor()
        return self.cur

    def __exit__(self, exc_type, exc_val, exc_tb):
        self.cur.close()
        self.conn.close()

with DBConnection('192.168.121.xxx', user="xxx", passwd="123456", db="xxx") as cur:
    cur.execute("select * from studnet;")
    result = cur.fetchall()
    print(result)

完成DBConnection这个类,每次连接数据库时,只要简单的调用with语句即可,不需要关心数据库的关闭、异常等。

上下文管理器的使用

#自定义一个上下文管理类
class MyOpen():
    def __init__(self):
        print("初始化方法")
    def __enter__(self):
        print("执行enter方法")
    def __exit__(self, exc_type, exc_val, exc_tb):
        print("执行exit方法")
print("===实例化对象====")
t= MyOpen()
print("===with语句====")
with MyOpen() as f:
    print("执行代码")
输出:
===实例化对象====
初始化方法
===with语句====
初始化方法
执行enter方法
执行代码
执行exit方法

在实例化对象时,并不会调用__enter__()方法,一般使用with语句,开启一个上下文环境,进入with语句块调用__enter__()方法,然后执行语句体,离开with语句块时,调用 __exit__() 方法。

上下文管理器的异常处理

根据上下文管理的原理,上下文管理器的原理是实现了__enter__()和 __exit__() 这两个方法,所以我们可以根据此原理来自定义自己的上下文管理器。

class MyOpen(object):
    """自定义上下文管理类"""

    def __init__(self, file, mode):
        self._file = file
        self._mode = mode

    def __enter__(self):
        self._handle = open(self._file, self._mode)
        return self._handle

    def __exit__(self, exc_type, exc_val, exc_tb):
        print('Type: ', exc_type)
        print('Value:', exc_val)
        print('TreacBack:', exc_tb)
        self._handle.close()
# 读的模式打开文件,进行写操作,不支持
with MyOpen('123.txt', 'r') as f:
    f.write('python')

img

当with中执行的语句发生异常时,异常信息会被发送到 __exit__()方法的参数中, __exit__() 方法有如下三个参数:

exc_type : 异常类型

exc_val : 异常值

exc_tb : 异常回溯追踪

这三个参数都与异常有关,with语句会把异常的exc_type ,exc_val 和exc_tb传递给 __exit__() 方法,它让__exit__() 方法来处理异常 ,如果__exit__()返回的是True,那么这个异常就被忽略,并按照我们定义的方式进行抛出。如果__exit__()返回的是True以外的任何东西,那么这个异常将被with语句抛出。

class MyOpen(object):
    """自定义上下文管理类"""

    def __init__(self, file, mode):
        self._file = file
        self._mode = mode

    def __enter__(self):
        self._handle = open(self._file, self._mode)
        return self._handle

    def __exit__(self, exc_type, exc_val, exc_tb):
        # print('Type: ', exc_type)
        # print('Value:', exc_val)
        # print('TreacBack:', exc_tb)
        self._handle.close()
        print("异常已被处理")
        return True

# 读的模式打开文件,进行写操作,不支持
with MyOpen('123.txt', 'r') as f:
    f.write('python')
    
输出:
异常已被处理

所以,with 语法不仅可以简化资源操作的后续清除操作,还可以代替 try/finally 进行异常处理。

“ 参考: https://blog.csdn.net/weixin_44514338/article/details/106529199 https://blog.csdn.net/mydistance/article/details/82730014 ”

更多Python上下文管理器知识,大家可以自行搜索学习,下面给大家推荐一个python入门的学习课程,希望对大家有帮助。

目录
相关文章
|
6月前
|
搜索推荐 Python
Python上下文管理器DIY指南:从入门到精通,轻松驾驭资源管理
【7月更文挑战第6天】Python的上下文管理器是资源管理的利器,简化文件操作、网络连接等场景。通过定义类及`__enter__`、`__exit__`方法,可自定义管理器,如示例中的`MyContextManager`,实现资源获取与释放。使用with语句,提升代码可读性和维护性,不仅用于基本资源管理,还可扩展到事务控制、自动重试等高级应用,让编程更加高效和灵活。
74 0
|
4月前
|
Python
Python的Virtualenv与Venv环境管理器
介绍Python的两种环境管理工具Virtualenv和venv,包括它们的安装、创建、激活、退出环境以及查看帮助信息的方法,同时对比了两者的特点和使用场景。
111 2
Python的Virtualenv与Venv环境管理器
|
4月前
|
Python
Python软件包及环境管理器conda实战篇
详细介绍了如何使用conda进行Python软件包管理及环境管理,包括查看、安装、卸载软件包,切换源,管理不同版本的Python环境,以及解决使用过程中可能遇到的错误。
162 2
Python软件包及环境管理器conda实战篇
|
5月前
|
安全 数据库连接 数据库
Python深度解析:上下文协议设计与应用技巧
在Python编程中,资源管理是一个常见且重要的问题。无论是文件操作、网络连接还是数据库事务,都需要确保资源在使用后能够正确地释放或恢复到初始状态。Python通过上下文管理器提供了一种优雅的方式来处理资源的获取与释放,使得代码更加简洁、安全。
|
6月前
|
监控 安全 数据库
逆天改命!用自定义上下文管理器,让你的Python代码效率飙升
【7月更文挑战第7天】Python上下文管理器简化资源管理,通过自定义实现优雅控制。使用with语句自动执行资源获取和释放,确保异常安全。例如,FileContextManager类通过__enter__打开文件,__exit__关闭并处理异常。自定义上下文管理器可封装重复逻辑,增强功能如日志和监控,提升代码效率与质量。利用这一工具,代码更简洁、高效且易于维护。**
46 1
|
6月前
|
程序员 数据库连接 Python
解锁Python新姿势:上下文管理器的自定义技巧,让你的编程之路更顺畅
【7月更文挑战第7天】Python上下文管理器简化资源管理,确保异常时资源正确释放。通过实现`__enter__`和`__exit__`或使用`contextmanager`装饰器自定义管理器。示例展示了类定义和装饰器方法。自定义管理器提升代码可读性,防止资源泄露,是高效编程的关键。**
56 1
|
6月前
|
安全 数据库连接 Python
告别繁琐!自定义Python上下文管理器,让你的资源管理变得如此简单
【7月更文挑战第4天】在Python中,上下文管理器通过`with`语句简化资源管理,确保资源的自动获取与释放,增强程序稳定性。自定义上下文管理器依靠`__enter__`和`__exit__`方法,例如,`CustomFileManager`类展示了如何记录文件操作。自定义管理器能简化代码、保证资源安全释放和提供定制逻辑,从而提升代码的健壮性和可维护性。它是处理文件、连接等资源管理的强大工具。
51 2
|
6月前
|
Python
告别混乱!Python上下文管理器的自定义实践,让你的代码更加整洁有序
【7月更文挑战第6天】Python的上下文管理器通过`__enter__`和`__exit__`方法自动处理资源的获取与释放,如文件操作。使用with语句结合自定义类(如`FileManager`示例),能确保文件在使用后正确关闭,防止资源泄漏,提升代码整洁度和健壮性。自定义上下文管理器是代码模块化和错误处理的好实践。
40 0
|
6月前
|
数据采集 程序员 Python
深度定制Python上下文管理器,让你的代码世界从此井井有条
【7月更文挑战第6天】Python的上下文管理器简化了资源管理,通过`__enter__`和`__exit__`方法自动处理获取和释放。例如,一个自定义的LoggingContextManager类在`__enter__`中配置日志并返回记录器,在`__exit__`中关闭文件。使用`with`语句,可以优雅地控制日志文件的生命周期,提高代码的整洁性和健壮性。
45 0
|
6月前
|
Python
惊呆了!学会这一招,你的Python上下文管理器也能玩出花样来文管理器也能玩出花样来
【7月更文挑战第6天】Python的上下文管理器是资源优雅管理的关键,与with语句结合,确保资源获取和释放。通过实现`__enter__`和`__exit__`,不仅能做资源分配和释放,还能扩展实现如计时、自动重试、事务处理等功能。例如,TimerContextManager类记录代码执行时间,展示了上下文管理器的灵活性。学习和利用这一机制能提升代码质量,增强功能,是Python编程的必备技巧。
40 0