contextmanager: 上下文管理器类和上下文管理器装饰器

原文

一. 什么是上下文管理器
二. 自定义一个上下文管理器类:
三. 使用contextmanager
四. 一个例子, sqlalchemy: 数据库的自动提交和回滚

一. 什么是上下文管理器

上下文管理器是在Python2.5之后加入的功能,可以在方便的需要的时候比较精确地分配和释放资源, with便是上下文管理器的最广泛的应用, 比如:

with open("test/test.txt","w") as f:
  f.write("hello")

这上会比使用try:...finally:f.close方便的多.

二. 自定义一个上下文管理器类:
class MyResource:
    # __enter__ 返回的对象会被with语句中as后的变量接受
    def __enter__(self):
        print('connect to resource')
        return self

    def __exit__(self, exc_type, exc_value, tb):
        print('close resource conection')

    def query(self):
        print('query data')

类中有两个特殊的魔术方法:
__enter__: with语句中的代码块执行前, 会执行__enter__, 返回的值将赋值给with句中as后的变量.
__exit__: with语句中的代码块执行结束或出错, 会执行_exit__

比如以下代码:

with Myresource() as r:
    r.query()

的打印结果为:

connect to resource
query data
close resource conection

那么有没有一个简化定义的方法呢, python提供了一个装饰器contextmanager

三. 使用contextmanager
from contextlib import contextmanager
class MyResource:
    def query(self):
        print('query data')

@contextmanager
def make_myresource():
    print('start to connect')
    yield MyResource()
    print('end connect')
    pass

被装饰器装饰的函数分为三部分:
with语句中的代码块执行前执行函数中yield之前代码
yield返回的内容复制给as之后的变量
with代码块执行完毕后执行函数中yield之后的代码

比如下方代码:

with make_myresource() as r:
     r.query()

的结果为:


start to connect
query data
end connect
四. 一个例子, sqlalchemy: 数据库的自动提交和回滚

在编程中如果频繁的修改数据库, 一味的使用类似try:... except..: rollback() raise e其实是不太好的.

比如某一段的代码的是这样的:

   try:
        gift = Gift()
        gift.isbn = isbn
        ... 
        db.session.add(gift)
        db.session.commit()
    except Exception as e:
        db.session.rollback()
        raise e

为了达到使用with语句的目的, 我们可以重写db所属的类:

from flask_sqlalchemy import SQLAlchemy as _SQLALchemy
class SQLAlchemy(_SQLALchemy):
    @contextmanager
    def auto_commit(self):
        try:
            yield
            self.session.commit()
        except Exception as e:
            db.session.rollback()
            raise e

这时候, 在执行数据的修改的时候便可以:

 with db.auto_commit():
        gift = Gift()
        gift.isbn = isbndb.session.add(gift)
        db.session.add(gift)

with db.auto_commit():
    user = User()
    user.set_attrs(form.data)
    db.session.add(user)
  • 7
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
在Python中,上下文管理器是一种处理资源的方式,它能够自动分配并且释放资源,比如打开和关闭文件、数据库连接等。上下文管理器通常使用with语句来实现,保证了代码块执行完毕后,资源被释放。Python提供了两种实现上下文管理器的方式:使用和使用装饰器。 使用实现上下文管理器时,需要定义一个,并且实现`__enter__()`和`__exit__()`方法。`__enter__()`方法会在进入代码块前被调用,它负责资源的分配;`__exit__()`方法则会在代码块执行完成后被调用,负责资源的释放。 以下是一个简单的示例: ```python class MyContext: def __init__(self, resource): self.resource = resource def __enter__(self): print("Entering context...") return self.resource def __exit__(self, exc_type, exc_val, exc_tb): print("Exiting context...") self.resource.close() with MyContext(open("file.txt", "w")) as f: f.write("Hello World!") ``` 使用装饰器实现上下文管理器时,需要定义一个函数,并且使用`@contextlib.contextmanager`装饰器修饰它。这个函数需要使用`yield`语句将控制权传递给with语句块,然后在finally块中释放资源。 以下是一个简单的示例: ```python from contextlib import contextmanager @contextmanager def my_context(resource): try: print("Entering context...") yield resource finally: print("Exiting context...") resource.close() with my_context(open("file.txt", "w")) as f: f.write("Hello World!") ```
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值