使用 functools.wraps 定义函数修饰器

在 Python 中,函数修饰器是一种非常强大的工具,可以用来增强或修改函数的行为。然而,在定义函数修饰器时,如果不正确地处理函数的元数据(如名称、文档字符串等),可能会导致一些意外的问题。functools.wraps 是一个非常有用的装饰器,它可以确保修饰器正确地保留被修饰函数的元数据。

本文将详细介绍如何使用 functools.wraps 来定义函数修饰器,并展示几个实际应用的例子。

1. 理解 functools.wraps

functools.wraps 是一个装饰器,它用于更新包装函数的元数据,使其与被包装的原始函数一致。具体来说,它会更新以下元数据:

  • 函数名称 (__name__)
  • 函数文档字符串 (__doc__)
  • 函数注释 (__annotations__)
  • 函数模块 (__module__)
  • 函数属性字典 (__dict__)

2. 基本用法

首先来看一个简单的例子,不使用 functools.wraps 的情况。

python 复制代码
def simple_decorator(func):
    def wrapper(*args, **kwargs):
        print("Before function call")
        result = func(*args, **kwargs)
        print("After function call")
        return result
    return wrapper

@simple_decorator
def my_function():
    """This is a simple function."""
    print("Function called")

print(my_function.__name__)  # 输出: wrapper
print(my_function.__doc__)   # 输出: None

可以看到,my_function 的名称和文档字符串都被替换成了 wrapper 的元数据。接下来,我们使用 functools.wraps 来解决这个问题。

python 复制代码
from functools import wraps

def simple_decorator(func):
    @wraps(func)
    def wrapper(*args, **kwargs):
        print("Before function call")
        result = func(*args, **kwargs)
        print("After function call")
        return result
    return wrapper

@simple_decorator
def my_function():
    """This is a simple function."""
    print("Function called")

print(my_function.__name__)  # 输出: my_function
print(my_function.__doc__)   # 输出: This is a simple function.

现在,my_function 的名称和文档字符串都得到了正确的保留。

3. 实际应用

接下来,我们看几个实际的应用场景,展示如何使用 functools.wraps 来定义函数修饰器。

3.1 日志记录

python 复制代码
from functools import wraps

def log(func):
    @wraps(func)
    def wrapper(*args, **kwargs):
        print(f"Calling {func.__name__} with args: {args}, kwargs: {kwargs}")
        result = func(*args, **kwargs)
        print(f"Finished calling {func.__name__}")
        return result
    return wrapper

@log
def add(a, b):
    """Add two numbers and return the sum."""
    return a + b

result = add(5, 3)
print(result)          # 输出: 8
print(add.__name__)    # 输出: add
print(add.__doc__)     # 输出: Add two numbers and return the sum.

3.2 性能监控

python 复制代码
import time
from functools import wraps

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

@timeit
def heavy_computation(n):
    """Perform a heavy computation."""
    total = 0
    for i in range(n):
        total += i
    return total

result = heavy_computation(1000000)
print(result)          # 输出: 499999500000
print(heavy_computation.__name__)  # 输出: heavy_computation
print(heavy_computation.__doc__)   # 输出: Perform a heavy computation.

3.3 输入验证

python 复制代码
from functools import wraps

def validate_input(func):
    @wraps(func)
    def wrapper(a, b):
        if not isinstance(a, int) or not isinstance(b, int):
            raise TypeError("Inputs must be integers")
        return func(a, b)
    return wrapper

@validate_input
def multiply(a, b):
    """Multiply two numbers."""
    return a * b

result = multiply(5, 3)
print(result)          # 输出: 15
print(multiply.__name__)  # 输出: multiply
print(multiply.__doc__)   # 输出: Multiply two numbers.

总结

functools.wraps 是一个非常有用的工具,可以确保函数修饰器正确地保留被修饰函数的元数据。通过使用 functools.wraps,我们可以避免因元数据丢失而导致的各种问题,使代码更加健壮和易读。

以上几个例子展示了如何使用 functools.wraps 来定义常见的函数修饰器,包括日志记录、性能监控和输入验证。希望这些示例能够帮助你在实际开发中更好地利用函数修饰器。

相关推荐
凹凸曼说我是怪兽y26 分钟前
Redis分布式锁详细实现演进与Redisson深度解析
数据库·redis·分布式
霍理迪28 分钟前
CSS——背景样式以及雪碧图、渐变
前端·css
TTGGGFF29 分钟前
Supertonic 部署与使用全流程保姆级指南(附已部署镜像)
开发语言·python
love530love33 分钟前
升级到 ComfyUI Desktop v0.7.0 版本后启动日志报 KeyError: ‘tensorrt‘ 错误解决方案
开发语言·windows·python·pycharm·virtualenv·comfyui·comfyui desktop
jump_jump4 小时前
基于 Squoosh WASM 的浏览器端图片转换库
前端·javascript·性能优化
落羽凉笙7 小时前
Python学习笔记(3)|数据类型、变量与运算符:夯实基础,从入门到避坑(附图解+代码)
笔记·python·学习
Quintus五等升7 小时前
深度学习①|线性回归的实现
人工智能·python·深度学习·学习·机器学习·回归·线性回归
天远Date Lab7 小时前
Python实战:对接天远数据手机号码归属地API,实现精准用户分群与本地化运营
大数据·开发语言·python
@淡 定7 小时前
Redis热点Key独立集群实现方案
数据库·redis·缓存
小二·7 小时前
前端监控体系完全指南:从错误捕获到用户行为分析(Vue 3 + Sentry + Web Vitals)
前端·vue.js·sentry