Python 装饰器 @wraps()

装饰器(Decorator)

在 Python 中,装饰器是一个特殊的函数,它可以修改或扩展另一个函数的行为。装饰器是一个小函数,它接受另一个函数作为参数,并返回一个新的函数,该新的函数“包装”了原始函数。

元数据丢失问题

当你创建一个装饰器时,你实际上是创建了一个新的函数,该函数调用原始函数。然而,这个新的函数不会继承原始函数的元数据(例如名称、文档字符串、参数)。这可能会导致问题,例如使用 help() 或 inspect 模块来检查函数的元数据时。

wraps 的作用

wraps 装饰器来自 functools 模块,它可以解决元数据丢失的问题。当你使用 wraps 来装饰一个装饰器时,它将确保原始函数的元数据被复制到新的函数中。

让我们来探讨如果不使用 @wraps() 装饰器时会发生什么情况。😊

不使用 @wraps() 的结果

如果不使用 @wraps() 装饰器,原始函数的元数据(例如函数名、文档字符串、参数签名)将不被保留。这可能会导致以下问题:

  1. 函数名和文档字符串丢失:装饰后的函数将具有不同的函数名和文档字符串,这可能会使函数的目的和行为不明确。
  2. 参数签名不正确:装饰后的函数的参数签名将不匹配原始函数的参数签名,这可能会导致调用函数时的错误。
  3. 调试困难:当错误发生时,调试器将显示装饰后的函数名和签名,而不是原始函数的名称和签名,使得错误的来源难以确定。

代码示例

def MyDecorator(func):
    def wrapper(*args, **kwargs):
        print("函数调用前发生了什么。")
        result = func(*args, **kwargs)
        print("函数调用后发生了什么。")
        return result
    return wrapper

@MyDecorator
def add(a, b):
    """加两个数字"""
    return a + b

print(add.__name__)  # 输出:wrapper
print(add.__doc__)  # 输出:None
help(add)  # 输出:没有文档可用

如你所见,装饰后的函数 丢失了其原始函数名和文档字符串。这可能会导致代码维护和调试困难。

使用 @wraps() 的结果

相比之下,如果使用 @wraps() 装饰器,原始函数的元数据将被保留,避免了上述问题。

from functools import wraps

def MyDecorator(func):
    @wraps(func)
    def wrapper(*args, **kwargs):
        print("函数调用前发生了什么。")
        result = func(*args, **kwargs)
        print("函数调用后发生了什么。")
        return result
    return wrapper

@MyDecorator
def add(a, b):
    """加两个数字"""
    return a + b

print(add.__name__)  # 输出:add
print(add.__doc__)  # 输出:加两个数字
help(add)  # 输出:正确的文档

使用 @wraps() 后,装饰后的函数 add 保留了其原始函数名和文档字符串,避免了代码维护和调试困难。

  • 4
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
@wraps是functools模块中的一个装饰器函数,它的作用是用来修饰一个装饰器函数,使得被修饰的函数保留原来的函数名和文档字符串。当一个函数被修饰时,它的__name__属性将变成装饰器函数的名字,而不是被修饰函数的名字。同时,它还会将被修饰函数的__doc__属性(即文档字符串)也赋值给修饰后的函数。 在引用中的示例中,使用了@wraps修饰器来装饰my_decorator函数内部的wrapper函数。这样一来,当使用my_decorator装饰test函数时,不仅保留了test函数的名字和文档字符串,还将装饰器函数的名字和文档字符串赋值给了被修饰后的函数。 而在引用中的示例中,未使用@wraps修饰器,所以被修饰的函数test在被调用后,其__name__属性变成了wrapper,而不是test,同时__doc__属性也变成了装饰器函数的文档字符串,而不是被修饰函数的文档字符串。 通过使用@wraps修饰器,可以解决被修饰函数的一些属性被覆盖的问题,保证被修饰函数在被调用时能够保留原有的属性。这在一些需要对函数进行装饰的场景中非常有用。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* [Python @wraps的用法](https://blog.csdn.net/weixin_42596275/article/details/125967916)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *3* [python被修饰的函数消失问题解决(基于wraps函数)](https://download.csdn.net/download/weixin_38611812/12858449)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值