登录
首页 >  文章 >  python教程

Python装饰器参数如何通过inspect模块获取?

时间:2024-11-26 19:18:53 474浏览 收藏

各位小伙伴们,大家好呀!看看今天我又给各位带来了什么文章?本文标题《Python装饰器参数如何通过inspect模块获取?》,很明显是关于文章的文章哈哈哈,其中内容主要会涉及到等等,如果能帮到你,觉得很不错的话,欢迎各位多多点评和分享!

Python装饰器参数如何通过inspect模块获取?

通过inspect获取装饰器参数的难题

在python中,inspect模块提供了获取函数元数据的方法。然而,获取装饰器传入的参数却并非易事。

以以下代码为例:

import inspect
from functools import wraps

def task(_id, params):
    
    def decorator(func):
        @wraps(func)
        def wrapper(*args, **kwargs):
            return func(*args, **kwargs)

        return wrapper

    return decorator

@task(_id="aaa", params=(1,2,3))
def foo(q=1):
    ...

if __name__ == '__main__':
    print(inspect.getclosurevars(foo.__wrapped__).nonlocals)  # 不生效,期望获取(_id="aaa", params=(1,2,3))信息

使用inspect.getclosurevars获取闭包变量不起作用,因为装饰器内部定义的变量不在foo.__wrapped__的闭包中。

解决方案

inspect无法直接获取装饰器参数,目前有以下变通方案:

1. 在装饰器内部存储参数

在decorator函数中显式存储参数,然后在包装器函数中使用它们。

2. 使用ast解析代码

使用ast模块解析装饰器的代码,并提取参数。不过,这种方法仅适用于特定类型的代码,缺乏通用性。

示例代码:

import ast

def get_decorator_args(func):
    source = inspect.getsource(func)
    tree = ast.parse(source)

    for node in ast.walk(tree):
        if isinstance(node, ast.FunctionDef):
            for decorator in node.decorator_list:
                if isinstance(decorator, ast.Call) and decorator.func.id == "task":
                    args = {
                        arg.arg: ast.literal_eval(arg.value)
                        for arg in decorator.keywords
                    }
                    return args
    return {}

print(get_decorator_args(foo))  

# {'_id': 'aaa', 'params': (1, 2, 3)}

需要注意的是,该解决方案仅适用于上述特定代码片段。对于更复杂的装饰器,可能需要定制化的解析代码。

到这里,我们也就讲完了《Python装饰器参数如何通过inspect模块获取?》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于的知识点!

相关阅读
更多>
最新阅读
更多>
课程推荐
更多>