编写高质量Python (第26条) 用 functools.wraps 定义函数装饰器

2023-12-03 08:28

本文主要是介绍编写高质量Python (第26条) 用 functools.wraps 定义函数装饰器,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

第26条 用 functools.wraps 定义函数装饰器

​ Python 中有一个特殊写法,可以用装饰器来封装某个函数,从而让函数在执行这个函数之前与执行完这个函数之后,分别运行某些代码。这意味着,调用者传给参数的参数值、函数返回给调用者的值,以及函数抛出的异常,都可以由装饰器访问并修改。这是个很有用 的机制,能够保证用户以正确的方式使用函数,也能用来调试程序或实现函数注册功能,此外还有很多用途。

​ 例如,假设我们要把函数执行时收到的参数与返回的值记录下来。这在调试递归函数是很有用的,因为我们要知道,这个函数执行每一层递归时,输入的是什么参数,返回的是什么值。下面我们就定义这样一个装饰器,在实现这个修饰器时,用 *args 与 **kwargs 表示受修饰的原函数 func 所收到的参数(参见 第22条和第23条)。

def trace(func):def wrapper(*args, **kwargs):result = func(*args, **kwargs)print(f'{func.__name__}({args!r},{kwargs!r})'f' -> {result!r}')return resultreturn wrapper

​ 写好之后,我们用 @ 符号把修饰器运用在想要调试的函数上面。

@trace
def fibonacci(n):"""Return the n-th Fibonacci number """if n in (0, 1):return nreturn (fibonacci(n - 2) + fibonacci(n - 1))

​ 这样写,相当于先把受修饰的函数传给修饰器,然后将修饰器所返回的值赋给原来那个函数。这样的话,如果我们继续通过原来的那个名字调用函数,那么执行的就是修饰之后的函数。

fibonacci = trace(fibonacci)

​ 修饰过的 fibonacci 函数,会在执行自身的代码之前,先执行 wrapper 里位于 func(*args, **kwargs) 那一行之前的逻辑;并且在执行完自身的代码后,执行 wrapper 里位于 func (*args, **kwargs) 那一行之后的逻辑。本例中,它会在执行完自身的代码之后,打印这次执行所用的参数与返回值,这样就能看到整个递归栈的情况了。

fibonacci(4)>>>
fibonacci((0,),{}) -> 0
fibonacci((1,),{}) -> 1
fibonacci((2,),{}) -> 1
fibonacci((1,),{}) -> 1
fibonacci((0,),{}) -> 0
fibonacci((1,),{}) -> 1
fibonacci((2,),{}) -> 1
fibonacci((3,),{}) -> 2
fibonacci((4,),{}) -> 3

​ 这样写确实能满足需求,但是会带来一个我们不愿意看到的副作用。修饰器返回的那个值,也就是刚才调用的 fibonacci ,它的名字并不叫 “fibonacci”。

print(fionacci)>>>
<function trace.<locals>.wrapper at 0x1090ad120>

​ 这种现象解释起来并不困难。trace 函数返回的,是它里面定义的 wrapper 函数,所以,当我们把这个返回值赋给 fibonacci 之后,fibonacci 这个名称所表示的自然就是 wrapper 了。问题在于,这样可能会干扰那些想要利用 introspection 机制来运作的工具,例如调试器(debugger)(参见 第80条)。

​ 例如,如果用内置的 help 函数来查看修饰后的 fibonacci,那么打印出来的并不是我们想看的帮助文档,他本来该打印前面定义时写的那行 'Return the n-th Fibonacci number’文本才对。

help(fabonacci)>>>
Help on function wrapper in module __main__:wrapper(*args, **kwargs)

​ 对象序列化器(object serializer, 参见 第68条) 也无法正常运作,因为它不能确定受修饰函数的位置。

import pickle
pickle.dumps(fibonacci)>>>
Traceback ...
AttributeError: Can't pickle local object 'trace.<locals>.wrapper'

​ 要像解决这些问题,可以改用 functools 内置模块之中的 wraps 辅助函数来实现。wraps 本身也是个修饰器,它可以帮助你编写自己的装饰器。把它运作到 wrapper 函数上面,它就会将重要的元数据 (metadata) 全都从内部函数复制到外部函数。

from functools import wrapsdef trace(func):@wraps(func)def wrapper(*args, **kwargs):result = func(*args, **kwargs)print(f'{func.__name__}({args!r},{kwargs!r})'f' -> {result!r}')return resultreturn wrapper@trace
def fibonacci(n):"""Return the n-th Fibonacci number """if n in (0, 1):return nreturn (fibonacci(n - 2) + fibonacci(n - 1))

​ 现在我们就可以通过 help 函数看到正确的文档了。虽然原来的 fibonacci 函数现在封装在装饰器上,但我们还可以看到它的文档。

help(fibonacci)>>>
Help on function fibonacci in module __main__:fibonacci(n)Return the n-th Fibonacci number

​ 对象序列化,也正常了。

print(pickle.dumps(fibonacci))>>>
b'\x80\x04\x95\x1a\x00\x00\x00\x00\x00\x00\x00\x8c\x08__main__\x94\x8c\tfibonacci\x94\x93\x94.'

​ 除了这里讲到的几个方面之外,Python 函数还有很多属性(例如 __name__,_moudle_,__annotations__)也应该在受到封装时得以保留,这样才能让相关的接口正常运作。wraps 可以帮助保存这些属性,使程序表现出正确的行为。

这篇关于编写高质量Python (第26条) 用 functools.wraps 定义函数装饰器的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/448768

相关文章

MySQL常用字符串函数示例和场景介绍

《MySQL常用字符串函数示例和场景介绍》MySQL提供了丰富的字符串函数帮助我们高效地对字符串进行处理、转换和分析,本文我将全面且深入地介绍MySQL常用的字符串函数,并结合具体示例和场景,帮你熟练... 目录一、字符串函数概述1.1 字符串函数的作用1.2 字符串函数分类二、字符串长度与统计函数2.1

Python实现网格交易策略的过程

《Python实现网格交易策略的过程》本文讲解Python网格交易策略,利用ccxt获取加密货币数据及backtrader回测,通过设定网格节点,低买高卖获利,适合震荡行情,下面跟我一起看看我们的第一... 网格交易是一种经典的量化交易策略,其核心思想是在价格上下预设多个“网格”,当价格触发特定网格时执行买

Python标准库之数据压缩和存档的应用详解

《Python标准库之数据压缩和存档的应用详解》在数据处理与存储领域,压缩和存档是提升效率的关键技术,Python标准库提供了一套完整的工具链,下面小编就来和大家简单介绍一下吧... 目录一、核心模块架构与设计哲学二、关键模块深度解析1.tarfile:专业级归档工具2.zipfile:跨平台归档首选3.

使用Python构建智能BAT文件生成器的完美解决方案

《使用Python构建智能BAT文件生成器的完美解决方案》这篇文章主要为大家详细介绍了如何使用wxPython构建一个智能的BAT文件生成器,它不仅能够为Python脚本生成启动脚本,还提供了完整的文... 目录引言运行效果图项目背景与需求分析核心需求技术选型核心功能实现1. 数据库设计2. 界面布局设计3

Python进行JSON和Excel文件转换处理指南

《Python进行JSON和Excel文件转换处理指南》在数据交换与系统集成中,JSON与Excel是两种极为常见的数据格式,本文将介绍如何使用Python实现将JSON转换为格式化的Excel文件,... 目录将 jsON 导入为格式化 Excel将 Excel 导出为结构化 JSON处理嵌套 JSON:

Python操作PDF文档的主流库使用指南

《Python操作PDF文档的主流库使用指南》PDF因其跨平台、格式固定的特性成为文档交换的标准,然而,由于其复杂的内部结构,程序化操作PDF一直是个挑战,本文主要为大家整理了Python操作PD... 目录一、 基础操作1.PyPDF2 (及其继任者 pypdf)2.PyMuPDF / fitz3.Fre

python设置环境变量路径实现过程

《python设置环境变量路径实现过程》本文介绍设置Python路径的多种方法:临时设置(Windows用`set`,Linux/macOS用`export`)、永久设置(系统属性或shell配置文件... 目录设置python路径的方法临时设置环境变量(适用于当前会话)永久设置环境变量(Windows系统

python中列表应用和扩展性实用详解

《python中列表应用和扩展性实用详解》文章介绍了Python列表的核心特性:有序数据集合,用[]定义,元素类型可不同,支持迭代、循环、切片,可执行增删改查、排序、推导式及嵌套操作,是常用的数据处理... 目录1、列表定义2、格式3、列表是可迭代对象4、列表的常见操作总结1、列表定义是处理一组有序项目的

python运用requests模拟浏览器发送请求过程

《python运用requests模拟浏览器发送请求过程》模拟浏览器请求可选用requests处理静态内容,selenium应对动态页面,playwright支持高级自动化,设置代理和超时参数,根据需... 目录使用requests库模拟浏览器请求使用selenium自动化浏览器操作使用playwright

python使用try函数详解

《python使用try函数详解》Pythontry语句用于异常处理,支持捕获特定/多种异常、else/final子句确保资源释放,结合with语句自动清理,可自定义异常及嵌套结构,灵活应对错误场景... 目录try 函数的基本语法捕获特定异常捕获多个异常使用 else 子句使用 finally 子句捕获所