详细分析Python中的@wraps(func)基本知识(附Demo)

本文介绍了Python装饰器中的@wraps函数的重要性,它能帮助保留被装饰函数的元数据,如名称和文档字符串。通过示例展示了使用和不使用@wraps装饰器对函数影响的对比,强调了其在保持代码可读性和调试性方面的价值。
摘要由CSDN通过智能技术生成

前言

  1. 对于装饰器的基本知识推荐阅读:详细分析Python装饰器(附Demo)
  2. 对于*args, **kwargs的理解:Python关于 *args 和 **kwargs 参数的详解(全)

没有使用@wraps(func)的Demo:

def my_decorator(func):
    def wrapper(*args, **kwargs):
        """这是装饰器内部的包装函数"""
        print("Before calling the function")
        result = func(*args, **kwargs)
        print("After calling the function")
        return result
    return wrapper

@my_decorator
def say_hello():
    """这是原始的 say_hello 函数"""
    print("Hello!")

print(say_hello.__name__)  # 输出包装函数的名称,而不是原始函数的名称
print(say_hello.__doc__)   # 输出包装函数的文档字符串,而不是原始函数的文档字符串

截图如下:

在这里插入图片描述

未使用 @wraps(func) 装饰器时,被装饰的函数会丢失原始函数的元数据,包括函数名和文档字符串

1. 基本知识

在Python中,@wraps(func) 是一个装饰器,用于帮助保留被装饰函数的元数据。它实际上是一个装饰器工具,来自于 functools 模块,用于解决使用装饰器时可能出现的一些问题,比如丢失被装饰函数的名称、文档字符串、参数签名等

@wraps(func) 装饰器的作用包括

  • 保留函数元数据: 装饰器在包装被装饰函数时会改变函数的一些属性,例如函数名、文档字符串等。使用 @wraps(func) 装饰器可以确保包装后的函数保留原始函数的元数据,使得调试、文档生成等更加方便

  • 提高可读性: 在使用装饰器时,如果不使用 @wraps(func) 装饰器,可能会导致装饰后的函数失去原函数的一些信息,这会降低代码的可读性。通过使用 @wraps(func) 装饰器,可以使得装饰后的函数更具可读性,更接近原始函数

2. Demo

结合前言给出的Demo,看如下Demo(保留元数据):

from functools import wraps

def my_decorator(func):
    @wraps(func)  # 使用 @wraps 装饰器
    def wrapper(*args, **kwargs):
        """这是装饰器内部的包装函数"""
        print("Before calling the function")
        result = func(*args, **kwargs)
        print("After calling the function")
        return result
    return wrapper

@my_decorator
def say_hello():
    """这是原始的 say_hello 函数"""
    print("Hello!")

print(say_hello.__name__)  # 输出原始函数的名称
print(say_hello.__doc__)   # 输出原始函数的文档字符串

截图如下:

在这里插入图片描述
总的来说:

使用了 @wraps(func) 装饰器,输出的函数名和文档字符串分别是原始函数的名称和文档字符串

再附上相关的Demo:

from functools import wraps

def log_parameters_with_wraps(func):
    @wraps(func)  # 使用 @wraps 装饰器
    def wrapper(*args, **kwargs):
        """这是装饰器内部的包装函数"""
        print(f"Arguments: {args}, Keyword arguments: {kwargs}")
        return func(*args, **kwargs)
    return wrapper

@log_parameters_with_wraps
def multiply(x, y):
    """这是原始的 multiply 函数"""
    return x * y

# 调用 multiply 函数并传递关键字参数
multiply(10, y=20)


print(multiply.__name__)  # 输出函数名,应该是原始函数的名称
print(multiply.__doc__)   # 输出函数文档字符串,应该是原始函数的文档字符串

截图如下:

在这里插入图片描述

  • 11
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
@wrapsfunctools模块的一个装饰器函数,它的作用是用来修饰一个装饰器函数,使得被修饰的函数保留原来的函数名和文档字符串。当一个函数被修饰时,它的__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
发出的红包

打赏作者

码农研究僧

你的鼓励将是我创作的动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值