详解 Python 中的 @wraps 装饰器

5a70d9aa4a87f12bd4650bbf6c1f4c3c.png

更多Python学习内容:ipengtao.com

在Python编程中,装饰器(decorators)是一种非常强大的工具,可以在不修改原有函数代码的前提下,扩展函数的功能。而在编写装饰器时,functools模块中的@wraps装饰器是一个不可或缺的工具。本文将详细介绍@wraps的作用和使用方法,并通过示例代码展示其实际应用。

一、什么是装饰器?

在深入了解@wraps之前,我们先简单介绍一下装饰器。装饰器本质上是一个高阶函数,它接受一个函数作为参数,并返回一个新函数。装饰器常用于在函数执行前后添加额外的行为,如日志记录、访问控制、缓存等。

示例代码:

def my_decorator(func):
    def wrapper():
        print("Something is happening before the function is called.")
        func()
        print("Something is happening after the function is called.")
    return wrapper

@my_decorator
def say_hello():
    print("Hello!")

say_hello()

在这个示例中,my_decorator是一个装饰器,它在目标函数say_hello执行前后打印额外的信息。

二、@wraps的作用

当我们使用装饰器时,装饰后的函数实际上是包装器函数(wrapper function),而不是原来的函数。这会导致一些问题,例如:

  1. 原函数的名称和文档字符串(docstring)被包装器函数覆盖。

  2. 原函数的属性(如__name____doc__)丢失。

为了解决这些问题,Python提供了functools.wraps装饰器,它可以帮助我们保留原函数的元数据。

示例代码:

from functools import wraps

def my_decorator(func):
    @wraps(func)
    def wrapper(*args, **kwargs):
        print("Something is happening before the function is called.")
        result = func(*args, **kwargs)
        print("Something is happening after the function is called.")
        return result
    return wrapper

@my_decorator
def say_hello():
    """This is the docstring of say_hello function."""
    print("Hello!")

print(say_hello.__name__)  # 输出: say_hello
print(say_hello.__doc__)   # 输出: This is the docstring of say_hello function.

在这个示例中,使用@wraps装饰wrapper函数后,say_hello函数的__name____doc__属性得到了保留。

三、@wraps的使用方法

@wrapsfunctools模块中的一个装饰器,它接受一个函数作为参数,并返回一个装饰器。@wraps通常用于装饰包装器函数,以便将原函数的元数据复制到包装器函数中。

基本用法:

from functools import wraps

def my_decorator(func):
    @wraps(func)
    def wrapper(*args, **kwargs):
        # 执行装饰器逻辑
        return func(*args, **kwargs)
    return wrapper

使用示例:

以下是一个更复杂的示例,展示如何使用@wraps在记录函数执行时间的装饰器中保留原函数的元数据。

import time
from functools import wraps

def timing_decorator(func):
    @wraps(func)
    def wrapper(*args, **kwargs):
        start_time = time.time()
        result = func(*args, **kwargs)
        end_time = time.time()
        print(f"Function {func.__name__} took {end_time - start_time:.4f} seconds to execute")
        return result
    return wrapper

@timing_decorator
def slow_function():
    """A function that simulates a slow process."""
    time.sleep(2)
    print("Slow function executed")

slow_function()
print(slow_function.__name__)  # 输出: slow_function
print(slow_function.__doc__)   # 输出: A function that simulates a slow process.

在这个示例中,timing_decorator装饰器用于计算并打印目标函数的执行时间。通过使用@wraps装饰wrapper函数,我们保留了原函数slow_function__name____doc__属性。

四、@wraps的详细说明

@wraps实际上是functools.update_wrapper函数的一个简化版。update_wrapper函数用于将目标函数的指定属性复制到包装器函数中。默认情况下,这些属性包括__module____name____qualname____annotations____doc__

示例代码:

from functools import update_wrapper

def my_decorator(func):
    def wrapper(*args, **kwargs):
        # 执行装饰器逻辑
        return func(*args, **kwargs)
    update_wrapper(wrapper, func)
    return wrapper

使用update_wrapper函数可以实现与@wraps相同的效果。@wraps只是一个方便的简化版,它接受与update_wrapper相同的参数,并将这些参数传递给update_wrapper

参数说明:

  • wrapped:被包装的函数。

  • assigned:默认值为WRAPPER_ASSIGNMENTS,指定要复制的属性。

  • updated:默认值为WRAPPER_UPDATES,指定要更新的属性。

from functools import WRAPPER_ASSIGNMENTS, WRAPPER_UPDATES

def custom_wraps(wrapped, assigned=WRAPPER_ASSIGNMENTS, updated=WRAPPER_UPDATES):
    def decorator(wrapper):
        update_wrapper(wrapper, wrapped, assigned, updated)
        return wrapper
    return decorator

五、实际应用场景

1. 缓存结果

使用装饰器缓存函数的结果,避免重复计算:

from functools import wraps

def memoize(func):
    cache = {}
    @wraps(func)
    def wrapper(*args):
        if args not in cache:
            cache[args] = func(*args)
        return cache[args]
    return wrapper

@memoize
def fibonacci(n):
    if n < 2:
        return n
    return fibonacci(n - 1) + fibonacci(n - 2)

print(fibonacci(10))  # 输出: 55
print(fibonacci.__name__)  # 输出: fibonacci
print(fibonacci.__doc__)   # 输出: None

2. 权限检查

在Web应用中,使用装饰器进行权限检查:

from functools import wraps

def require_permission(permission):
    def decorator(func):
        @wraps(func)
        def wrapper(*args, **kwargs):
            if not has_permission(permission):
                raise PermissionError("Permission denied")
            return func(*args, **kwargs)
        return wrapper
    return decorator

def has_permission(permission):
    # 假设有一个权限检查的逻辑
    return permission == "admin"

@require_permission("admin")
def admin_area():
    """This is the admin area."""
    print("Welcome to the admin area")

try:
    admin_area()
except PermissionError as e:
    print(e)

print(admin_area.__name__)  # 输出: admin_area
print(admin_area.__doc__)   # 输出: This is the admin area.

总结

在Python中,@wraps是一个非常有用的工具,特别是在编写装饰器时。它可以帮助我们保留原函数的元数据,确保代码的可读性和可维护性。通过本文的介绍和示例代码,应该已经掌握了@wraps的基本用法和实际应用。

如果你觉得文章还不错,请大家 点赞、分享、留言 下,因为这将是我持续输出更多优质文章的最强动力!


如果想要系统学习Python、Python问题咨询,或者考虑做一些工作以外的副业,都可以扫描二维码添加微信,围观朋友圈一起交流学习。

f15c6f0241188f726d5dc497571b96d9.gif

我们还为大家准备了Python资料和副业项目合集,感兴趣的小伙伴快来找我领取一起交流学习哦!

d8538e68af5f829500dd22ddadf53ce8.jpeg

往期推荐

历时一个月整理的 Python 爬虫学习手册全集PDF(免费开放下载)

Python基础学习常见的100个问题.pdf(附答案)

学习 数据结构与算法,这是我见过最友好的教程!(PDF免费下载)

Python办公自动化完全指南(免费PDF)

Python Web 开发常见的100个问题.PDF

肝了一周,整理了Python 从0到1学习路线(附思维导图和PDF下载)

@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、付费专栏及课程。

余额充值