当前位置: 首页 > news >正文

Python基础总结之functools.wraps介绍与应用

Python基础总结之functools.wraps介绍与应用

在Python编程中,装饰器(decorator)是一种非常强大的工具,它允许开发者在不改变函数本身的情况下,动态地增加函数的功能。使用装饰器时,常常会用到 functools.wraps,这个工具可以说是写装饰器的好帮手。本文将详细介绍 functools.wraps 的功能、作用,并通过一些示例展示它的实际应用。

functools.wraps 是什么?

functools.wraps 是Python标准库中的一个装饰器,位于 functools 模块内。它的主要作用是帮助开发者编写装饰器,使被装饰的函数保留原有的元信息(如函数名、文档字符串等)。使用 wraps 可以使得装饰器更透明,增强代码的可读性和可调试性。

为什么要使用 functools.wraps?

在编写装饰器时,如果不使用 functools.wraps,会导致一些问题,例如:

  1. 函数元信息丢失:装饰器会返回一个新的函数对象,这个新的函数对象通常会丢失原函数的名称、文档字符串和其他元信息。
  2. 调试困难:在调试代码时,缺少函数的元信息会使问题定位变得困难。

functools.wraps 通过将原函数的元信息复制到装饰器内部的包装函数上,解决了上述问题。

使用 functools.wraps 的示例

让我们来看一个简单的示例,展示如何在编写装饰器时使用 functools.wraps

示例一:没有使用 functools.wraps 的装饰器

def my_decorator(func):def wrapper(*args, **kwargs):print(f"Calling function {func.__name__}")return func(*args, **kwargs)return wrapper@my_decorator
def say_hello(name):"""Greet someone by their name."""return f"Hello, {name}!"print(say_hello.__name__)  # 输出:wrapper
print(say_hello.__doc__)   # 输出:None

在这个示例中,say_hello 函数被装饰器 my_decorator 装饰后,其名称和文档字符串都丢失了。

示例二:使用 functools.wraps 的装饰器

from functools import wrapsdef my_decorator(func):@wraps(func)def wrapper(*args, **kwargs):print(f"Calling function {func.__name__}")return func(*args, **kwargs)return wrapper@my_decorator
def say_hello(name):"""Greet someone by their name."""return f"Hello, {name}!"print(say_hello.__name__)  # 输出:say_hello
print(say_hello.__doc__)   # 输出:Greet someone by their name.

在这个示例中,使用了 @wraps(func),成功保留了原函数的名称和文档字符串。

应用场景

1. 日志记录

在需要记录函数调用日志时,可以使用 functools.wraps 来保留函数的原有信息,便于日志记录和调试。

from functools import wrapsdef log_decorator(func):@wraps(func)def wrapper(*args, **kwargs):print(f"Function {func.__name__} called with args: {args} and kwargs: {kwargs}")result = func(*args, **kwargs)print(f"Function {func.__name__} returned {result}")return resultreturn wrapper@log_decorator
def add(x, y):"""Add two numbers."""return x + yadd(2, 3)

2. 访问控制

在实现访问控制功能时,使用 functools.wraps 可以确保原函数的元信息不丢失,方便在装饰器内进行权限检查。

from functools import wrapsdef require_authentication(func):@wraps(func)def wrapper(user, *args, **kwargs):if not user.is_authenticated:raise PermissionError("User is not authenticated")return func(user, *args, **kwargs)return wrapperclass User:def __init__(self, name, authenticated):self.name = nameself.is_authenticated = authenticated@require_authentication
def get_user_data(user):"""Get user data if authenticated."""return f"User data for {user.name}"user = User("Alice", True)
print(get_user_data(user))

3. 异步编程

在异步编程中,functools.wraps 同样可以用于装饰异步函数,确保异步函数的元信息不丢失。

import asyncio
from functools import wrapsdef async_log_decorator(func):@wraps(func)async def wrapper(*args, **kwargs):print(f"Function {func.__name__} called with args: {args} and kwargs: {kwargs}")result = await func(*args, **kwargs)print(f"Function {func.__name__} returned {result}")return resultreturn wrapper@async_log_decorator
async def async_add(x, y):"""Asynchronously add two numbers."""await asyncio.sleep(1)  # 模拟异步操作return x + yasync def main():result = await async_add(2, 3)print(f"Result: {result}")asyncio.run(main())

总结

functools.wraps 是一个简洁而实用的工具,它在编写装饰器时起到了重要的作用,帮助我们保留原函数的元信息,增强代码的可读性和可维护性。无论是在日志记录、访问控制还是异步编程中,functools.wraps 都是一个不可或缺的利器。希望本文对你理解和使用 functools.wraps 能有所帮助。

http://www.lryc.cn/news/371038.html

相关文章:

  • UE5基础1-下载安装
  • 前端实现获取后端返回的文件流并下载
  • Windows下对于Qt中带 / 的路径的处理
  • [leetcode]swap-nodes-in-pairs
  • 国思RDIF.vNext全新低代码快速开发框架平台6.1版本发布(支持vue2、vue3)
  • 中国地市分布图
  • HCIA11 网络安全之本地 AAA 配置实验
  • 用Python处理Excel的资源
  • 2024年中国移动游戏市场研究报告
  • JS-12-es6常用知识-async
  • 使用winscp 通过中转机器(跳板机、堡垒机)密钥远程连接服务器,保姆级别教程
  • 力扣-1984. 学生分数的最小差值
  • 激动人心的LayerDiffusion终于可以在ComfyUI中使用了
  • 【JVM】finalize() 方法的定义与作用
  • 这10个前端库,帮我在工作中赢得了不少摸鱼时间!!
  • (2024最新)CentOS 7上在线安装MySQL 5.7
  • 【C++高阶】C++继承学习手册:全面解析继承的各个方面
  • 使用GPT-soVITS再4060下2小时训练声音模型以及处理断句带来的声音模糊问题
  • 如何对stm32查看IO功能。
  • docker构建jdk17镜像
  • Android Uri转File path路径,Kotlin
  • iOS界面设计要点:四大模块解析
  • 数字取证技术(Digital Forensics Technology)实验课II
  • Redis缓存的使用
  • 力扣LCP 08.剧情触发时间
  • Elasticsearch-IndexTemplate和DynamicTemplate 有什么区别
  • list集合自定义排序
  • PHP Cookies:应用与管理
  • 【GD32F303红枫派使用手册】第十四节 DAC-输出电压实验
  • java 使用Log4j进行日志记录