-1

我有一个装饰器,用于抑制和记录函数中的异常。代码是这样的:

def log_exceptions(func):
    def wrapper(*args, **kwargs):
        try:
            ret = func(*args, **kwargs)
        except Exception as e:
            print(e)
            print(args, kwargs)  # also log what arguments caused the exception
        return ret
    return wrapper

这里的一个问题是,很难手动将打印的参数值与函数的参数名称匹配,因为位置参数也可以进入内部kwargs,并且内部函数中也可能有argskwargs参数。args因此,将包装器中的andkwargs值与内部函数中的参数名称相匹配会很有用。

那么我的问题是,是否有内置的方法来进行这种匹配?如果不是,那么实现它的优雅方式是什么?

4

1 回答 1

1

如果可以确定签名,那么inspect.signature(func).bind(*args, **kwargs)就可以完成这项工作。

inspect.signature(func)(小写 s)尝试计算一个Signature表示可调用签名的对象(大写 S),结果Signaturebind方法将签名与提供的参数匹配,返回一个BoundArguments对象。结果对象的arguments属性是参数名称到值的有序映射,不包括依赖默认值的参数。

这是一个如何使用它的示例:

import inspect

try:
    sig = inspect.signature(func)
except (TypeError, ValueError):
    print('Could not determine function signature.')
else:
    try:
        bound = sig.bind(*args, **kwargs)
    except TypeError:
        print('Could not bind arguments to function signature.')
        print('Signature:', sig)
        print('Positional arguments:', args)
        print('Keyword arguments:', kwargs)
    else:
        # You can use bound.apply_defaults() to include defaults
        print('Bound arguments (defaults not included):')
        for name, value in bound.parameters:
            print('{}: {!r}'.format(name, value))
于 2020-01-20T23:17:53.323 回答