微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

如何使用上下文管理器装饰 Python 函数?

如何解决如何使用上下文管理器装饰 Python 函数?

给定一个现有的上下文管理器,我想要一个函数装饰器,使执行发生在 with 块内。

这应该使以下两个代码块等效:

@decorate(contextmanager)
async def f():
   ...

await f()
async def f():
    ...

async with contextmanager:
    await f()

这对于包装函数很方便,例如asyncio.Semaphore(n) 上下文。

标准库中是否已经存在像 decorate 这样的便捷实用程序?

解决方法

ContextDecorator,您可以创建继承自它的上下文管理器并像这样使用它

from functools import wraps
from contextlib import ContextDecorator

class MyContext(ContextDecorator):
    def __enter__(self):
        print('in enter')
        return self

    def __exit__(self,exc_type,exc_val,exc_tb):
        print('in exit')

some_context = MyContext()

@some_context
def foo(x,y):
    print(x + y)

foo(5,3)

但是,我不确定您是否可以将它用于您没有编写的现有内容,或者它是否适用于异步内容,在这种情况下,您可以编写这样的装饰器:

from functools import wraps

# Some context manager
class MyContext:
    def __enter__(self):
        print('in enter')
        return self

    def __exit__(self,exc_tb):
        print('in exit')

# Decotator that gets an existing context manager and uses it
def context_decorator(context_manager):

    def inner(func):
        @wraps(func)
        def wrapper(*args,**kwargs):
            with context_manager:
                return func(*args,**kwargs)

        return wrapper

    return inner

some_context = MyContext()

@context_decorator(some_context)
def foo(x,3)
in enter
8
in exit

ps。请注意,对于 asyncio.Semephore 之类的内容,您需要修改代码以在相关位置使用 async withawait

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其他元素将获得点击?
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。)
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbcDriver发生异常。为什么?
这是用Java进行XML解析的最佳库。
Java的PriorityQueue的内置迭代器不会以任何特定顺序遍历数据结构。为什么?
如何在Java中聆听按键时移动图像。
Java“Program to an interface”。这是什么意思?