编写高质量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 可以帮助保存这些属性,使程序表现出正确的行为。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/191556.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

深度学习(四):pytorch搭建GAN(对抗网络)

1.GAN 生成对抗网络&#xff08;GAN&#xff09;是一种深度学习模型&#xff0c;由两个网络组成&#xff1a;生成器&#xff08;Generator&#xff09;和判别器&#xff08;Discriminator&#xff09;。生成器负责生成假数据&#xff0c;而判别器则负责判断数据是真实的还是 f…

解决Linux的端口占用报错问题

文章目录 1 Linux报错2 解决方式 1 Linux报错 Port 6006 is in use. If a gradio.Blocks is running on the port, you can close() it or gradio.close_all(). 想起之前运行Gradio 6006&#xff0c;端口被占用 2 解决方式 输入 netstat -tpl查看当前一些端口号的占用号&a…

go第三方包发布(短精细)

1、清除其他依赖项 $ go mod tidy # 清除不必要的依赖依赖清除完成后&#xff0c;查看go.mod文件配置是否规范 module github.com/fyupeng/rpc-go-netty go 1.19 require ( )2、本地版本创建 $ git tag v0.1.0 # 本地 创建标签3、版本提交 $ git push github v0.1.0 # 推送…

Selector SelectionKey基础学习

netty技术内幕一(Selector,SelectionKey) Java Nio注意事项 # selector Selector类的使用(一) SelectionKey类的使用 /* package java.nio.channels;import java.io.Closeable; import java.io.IOException; import java.nio.channels.spi.SelectorProvider; import java.u…

面试就是这么简单,offer拿到手软(一)—— 常见非技术问题回答思路

面试系列&#xff1a; 面试就是这么简单&#xff0c;offer拿到手软&#xff08;一&#xff09;—— 常见非技术问题回答思路 面试就是这么简单&#xff0c;offer拿到手软&#xff08;二&#xff09;—— 常见65道非技术面试问题 文章目录 一、前言二、常见面试问题回答思路问…

cyclictest 交叉编译与使用

目录 使用版本问题编译 numactl编译 cyclictest使用参考 cyclictest 主要是用于测试系统延时&#xff0c;进而判断系统的实时性 使用版本 rt-tests-2.6.tar.gz numactl v2.0.16 问题 编译时&#xff0c;需要先编译 numactl &#xff0c;不然会有以下报错&#xff1a; arm-…

在 Linux 中使用 udev 规则固定摄像头节点

简介 通过编写 udev 规则来固定 USB 摄像头节点&#xff0c;以便在系统中始终使用相同的设备路径访问摄像头。 确定摄像头的供应商 ID 和产品 ID 使用 lsusb 命令确定连接的 USB 摄像头的供应商 ID 和产品 ID。示例命令及输出&#xff1a; $ lsusb Bus 001 Device 030: ID 220…

AI 绘画 | Stable Diffusion 电商模特

前言 上一篇文章讲到如何给人物更换背景和服装。今天主要讲电商模特,就是服装电商们的固定服装产品图片如何变成真人模特穿上的固定服装产品效果图。如果你掌握了 《AI 绘画 | Stable Diffusion 人物 换背景|换服装》,这篇文章对你来说,上手会更轻松。 教程 提取服装蒙版…

Java实现简单飞翔小鸟游戏

一、创建新项目 首先创建一个新的项目&#xff0c;并命名为飞翔的鸟。 其次在飞翔的鸟项目下创建一个名为images的文件夹用来存放游戏相关图片。 用到的图片如下&#xff1a;0~7&#xff1a; bg&#xff1a; column&#xff1a; gameover&#xff1a; ground&#xff1a; st…

记QListWidget中QPushButton QSS样式失效的“bug”

一、场景 有一个QListWidget的列表&#xff1b;里面存放了若干QListWidgetItem&#xff1b;每个QListWidgetItem与一个自定义类对象绑定——通过QListWidget的setItemWidget()实现。自定义对象继承于QWidget&#xff0c;且内含QPushButton。 二、bug描述 在该QListWidget的外…

Mybatis 分页查询的三种实现

Mybatis 分页查询 1. 直接在 sql 中使用 limit2. 使用 RowBounds3. 使用 Mybatis 提供的拦截器机制3.1 创建一个自定义拦截器类实现 Interceptor3.2 创建分页查询函数 与 sql3.3 编写拦截逻辑3.4 注册 PageInterceptor 到 Mybatis 拦截器链中3.5 测试 准备一个分页查询类 Data…

Clion调试QTQString看不到值问题处理

环境 Clion &#xff1a;2019.3.6 Qt &#xff1a;5.9.6&#xff08;MinGW&#xff09; 环境搭建参考&#xff1a;https://blog.csdn.net/qq_27953479/article/details/132338745 调试时QString看不到值问题处理 下载文件 qt.py : https://github.com/KDE/kdevelop/blob/…

CIS|安森美微光近红外增强相机论文解析

引言 在之前的文章中&#xff0c;我们介绍了索尼、安森美以及三星等Sensor厂家在车载领域中的技术论文&#xff0c;分析了各个厂家不同的技术路线、Sensor架构以及差异点。今天&#xff0c;笔者借豪威科技在移动端200Mega Pixels产品的技术论文&#xff0c;讲解消费级CIS传感器…

高级软件工程15本书籍

如果您想学习软件工程技能并提高您的专业知识&#xff0c;那么这里是您的最佳选择。我们有一本很棒的书&#xff0c;可以极大地增强您在软件工程方面的知识。 1&#xff09;干净的代码 Robert C. Martin 写了一本名为“干净代码&#xff1a;敏捷软件工艺手册”的书。在本书中&…

如何在WordPress中批量替换图片路径?

很多站长在使用WordPress博客或者搬家时&#xff0c;需要把WordPress文章中的图片路径进行替换来解决图片不显示的问题。总结一下WordPress图片路径批量替换的过程&#xff0c;方便有此类需求的站长们学习。 什么情况下批量替换图片路径 1、更换了网站域名 有许多网站建设初期…

面试 Java 基础八股文十问十答第二期

面试 Java 基础八股文十问十答第二期 作者&#xff1a;程序员小白条 ⭐点赞⭐收藏⭐不迷路&#xff01;⭐ 11.什么是反射&#xff1f;反射有哪些作用&#xff1f;反射在Sping中的体现 (1): 什么是反射? 反射可以在运行时获取到一个类的所有信息&#xff0c;包括(成员变量&am…

关于qiankun沙箱sandbox(面试题)

为什么要有js资源隔离机制&#xff1f; 主应用和子应用&#xff0c;相同的全局变量&#xff0c;可能会发生冲突&#xff0c;子应用和子应用之间&#xff0c;相同的全局变量&#xff0c;也可能会发生冲突。在这里我们主要指的就是window。 思路&#xff1a;打开沙箱时能够修改…

Spring中@Transactional注解

在Spring框架中&#xff0c;Transactional 是一个注解&#xff0c;用于声明事务性的方法。这个注解可以被应用在方法级别或类级别上。它提供了一种声明式的事务管理方式&#xff0c;避免了在代码中直接编写事务管理相关的代码。Transactional 注解能够将一个方法纳入到一个事务…

基于SSM的生鲜在线销售系统

末尾获取源码 开发语言&#xff1a;Java Java开发工具&#xff1a;JDK1.8 后端框架&#xff1a;SSM 前端&#xff1a;Vue 数据库&#xff1a;MySQL5.7和Navicat管理工具结合 服务器&#xff1a;Tomcat8.5 开发软件&#xff1a;IDEA / Eclipse 是否Maven项目&#xff1a;是 目录…

亚马逊云科技推出新一代自研芯片

北京——2023 年12月1日 亚马逊云科技在2023 re:Invent全球大会上宣布其自研芯片家族的两个系列推出新一代&#xff0c;包括Amazon Graviton4和Amazon Trainium2&#xff0c;为机器学习&#xff08;ML&#xff09;训练和生成式人工智能&#xff08;AI&#xff09;应用等广泛的工…