Python装饰器带括号和不带括号的理解

装饰器是 Python 中一个强大且灵活的特性,允许用户在不修改原有函数或类定义的基础上,为其增加额外功能。

今天在尝试自定义 Python 装饰器的时候遇到了一个问题,因为以前一直是使用装饰器,基本没有自定义过装饰器,所以写了一个不是常见的写法(符合装饰器的语法,但是用法上是错误的)

一、Python 装饰器的简单介绍

装饰器本质上是一个可调用对象(通常是函数),它接受一个函数作为输入,并返回一个新的函数作为输出。这个新函数通常会在执行原始函数前后添加额外的操作,从而扩展或改变原始函数的行为。

Python 提供了简洁的语法糖来应用装饰器,即在函数定义之前使用 @my_decorator 的格式。例如:

@my_decorator
def my_function():pass

这里的 @my_decorator 实际上是一个语法糖,会将 my_function 传递给 my_decorator 函数,并将返回的结果重新绑定到 my_function 上。因此上诉调用等同于:

def my_function():passmy_function = my_decorator(my_function)

这里需要强调一下,是将 my_function 传递给 @ 后面的整个部分,可以在看完文章之后再返回来理解一下这句话。

二、不带括号和带括号的 Python 装饰器

然后我就写了一个装饰器,不过我犯了一个错误,现在我把代码整理了一下贴了出来。第一种是通常的写法,第二种是我的写法,不过它的调用会有问题。不过这里目前可以看出区别就是一个不带括号,另一个带括号。

import timedef log1(func):def wrapper1():start = time.time()res = func()print("exec time: %.2f" % (time.time()-start))return resprint("log1 wrapper1")return wrapper1@log1
def func1():time.sleep(0.15)print("call func1")def log2():def wrapper2(func):start = time.time()res = func()print("exec time: %.2f" % (time.time()-start))return resprint("log2 wrapper2")return wrapper2@log2()  # 不带括号会报错,提示不需要参数,但是接收到了一个参数
def func2():time.sleep(0.15)print("call func2")if __name__ == '__main__':func1()# 如果不使用装饰器,那么:# func1 的调用方式等价于  func1 => log1(func1),这还是一个函数# func2 的调用方式等价于 func2 => log2()(func2),这不是一个函数,而是一个结果了

在这里插入图片描述

如果不使用装饰器语法,而是普通的 Python 代码的方式是这样的(结果同上):

import timedef log1(func):def wrapper1():start = time.time()res = func()print("exec time: %.2f" % (time.time()-start))return resprint("log1 wrapper1")return wrapper1def func1():time.sleep(0.15)print("call func1")def log2():def wrapper2(func):start = time.time()res = func()print("exec time: %.2f" % (time.time()-start))return resprint("log2 wrapper2")return wrapper2def func2():time.sleep(0.15)print("call func2")if __name__ == '__main__':func1 = log1(func1)func1()log2()(func2)

对于装饰器用法的代码,虽然没有调用 func2(),但是它就会直接执行,而且调用 func2() 会报错。

@log2()  # 不带括号会报错,提示不需要参数,但是接收到了一个参数
def func2():time.sleep(0.15)print("call func2")

在这里插入图片描述

不过这个错误反而提醒了我,装饰器的原理应该是:在程序运行时,会将被装饰的函数作为参数传递给装饰器,也就是 @ 后的整个部分。

因此对于 @log1 来说,这就相当于:func1 = log1(func1),所以之后执行 func1 就是被装饰后的函数了(可以观察到打印输出的语句)。函数是直接传到了装饰器内部,这样比较容易理解。

对于 @log2() 来说,这就相当于:log2()(func2),函数是传到了 log2() 返回的函数中了(这里是 wrapper2),这样它的结果就不是一个函数了(非 callable),而是一个具体的值了(这里的结果为 None)。如果我使用 @log2,那么就是把函数传到 log2 中,但是这个函数是无参数的,如果给它传递参数就会报错了。参数是传递给它返回的函数中,所以装饰器需要加上括号调用,即 @log2()。不过这里的写法犯了一个错误,因此导致了导致了它直接执行了,因为它返回的不是一个函数,而是它的执行结果了。所以,解决的方式就是在内部再嵌套一层函数,修改之后的代码如下:

def log2():def wrapper2(func):def wrapper3():start = time.time()res = func()print("exec time: %.2f" % (time.time()-start))return resreturn wrapper3print("log2 wrapper2")return wrapper2@log2()
def func2():time.sleep(0.15)print("call func2")if __name__ == '__main__':func2()# 等价于 func2 = log2()(func2) 这里它还是一个函数(可调用),而不是一个返回值了()

在这里插入图片描述

我认为这里最大的问题就是这个装饰器的语法糖实在是太便利的,这样对于使用是非常方便的,但是凡事有好处就有坏处,它反而不利于我们对它的理解了。特别是,我之前有过 Java 的注解使用和学习经验。如果只是简单的使用确实不需要理解它是怎么工作的,但是对于想要深入理解的同学来说,还是需要去了解背后的运行机制。这里要把握的一点就是:在运行时,会将被装饰的函数作为参数传递给 @ 后面这整个部分,如果不带括号就是直接作为参数传输传入,然后返回一个新的函数。如果带括号,就是传递给它的返回值(内层函数,所以内层函数还要再嵌套才行,不然就是直接执行函数了)。所以即使被装饰的函数在内层,它的执行也是没有问题的,但你要明白这个过程!

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

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

相关文章

vue3点击添加小狗图片,vue3拆分脚本

我悄悄蒙上你的眼睛 模板和样式 <template><div class"XueXi_Hooks"><img v-for"(dog, index) in dog1List" :src"dog" :key"index" /><button click"addDog1">点我添加狗1</button><hr …

WPF之工具栏菜单栏功能区。

1&#xff0c;菜单栏&#xff0c;工具栏&#xff0c;状态栏。 1.1&#xff0c;Menu中可添加菜单分隔条<Separator></Separator>作为分割线&#xff0c;使用Separator可以通过改变其template来自定义&#xff0c;Separator是无焦点的&#xff0c;如果简单的在MenuIt…

JavaScript异步编程——08-Promise的链式调用【万字长文,感谢支持】

前言 实际开发中&#xff0c;我们经常需要先后请求多个接口&#xff1a;发送第一次网络请求后&#xff0c;等待请求结果&#xff1b;有结果后&#xff0c;然后发送第二次网络请求&#xff0c;等待请求结果&#xff1b;有结果后&#xff0c;然后发送第三次网络请求。以此类推。…

【ARM 嵌入式 C 入门及渐进 1.2 -- 是否为 n 字节对齐】

文章目录 是否为 n 字节对齐 是否为 n 字节对齐 在C语言中&#xff0c;你可以定义一个宏来检查一个给定地址是否是n字节对齐的。这里的n应该是2的幂&#xff08;例如&#xff0c;2、4、8、16等&#xff09;。要做到这一点&#xff0c;可以利用位操作的特性。具体地&#xff0c…

品鉴中的平衡之美:如何欣赏红酒的口感与风格和谐

品鉴云仓酒庄雷盛红酒的过程&#xff0c;是对其口感与风格和谐的追求和欣赏。平衡是红酒品质的重要标志之一&#xff0c;它体现在红酒的色泽、香气、口感和余味等多个方面。通过欣赏红酒的平衡之美&#xff0c;我们可以更好地领略其精妙之处&#xff0c;感受其带来的美妙滋味。…

【Unity Animation 2D】Unity Animation 2D骨骼绑定与动画制作

一、图片格式为png格式&#xff0c;并且角色各部分分离 图片参数设置 需要将Sprite Mode设置为Single&#xff0c;否则图片不能作为一个整体 1、创建骨骼 1.1 旋转Create Bone&#xff0c;点击鼠标左键确定骨骼位置&#xff0c;移动鼠标再次点击鼠标左键确定骨骼&#xff0c…

大数据面试题第一期*4

题1、HDFS存储机制 &#xff08;1&#xff09;客户端向namenode请求上传文件 &#xff0c;namenode检查目标文件是否已存在 &#xff0c;父目录是否存在。 &#xff08;2&#xff09;namenode返回是否可以上传。 &#xff08;3&#xff09;客户端请求第一个 block上传到哪几个d…

Docker in Docker 的原理与实践

概述 Docker in Docker&#xff08;DinD&#xff09;是一个让 Docker 容器内可以运行另一个 Docker 沙箱环境的技术。常用于持续集成&#xff08;CI&#xff09;工作流程&#xff0c;其中需要构建和推送 Docker 镜像&#xff0c;而不污染主宿主机的 Docker 环境。 Docker in …

JS_jq选择器合集

基本选择器 选择器返回示范描述#id单个元素$("#test").class 集合元素 $(".test")element集合元素$("p") 所有p原素*集合元素$("*")所有原素select1,select2,selectn集合元素$("div,span,p.myClass")所有匹配原素 层次选择…

嵌入式C语言高级教程:实现基于STM32的智能健康监测手环

智能健康监测手环能够实时监控用户的生理参数&#xff0c;如心率、体温和活动量&#xff0c;对于健康管理和疾病预防非常有帮助。本教程将指导您如何在STM32微控制器上实现一个基本的智能健康监测手环。 一、开发环境准备 硬件要求 微控制器&#xff1a;STM32L476RG&#xf…

RS3236-3.3YUTDN4功能和参数介绍及PDF资料

RS3236-3.3YUTDN4功能和参数介绍及PDF资料-公司新闻-配芯易-深圳市亚泰盈科电子有限公司 品牌: RUNIC(润石) 封装: XDFN-4-EP(1x1) 描述: 带过温保护 输出类型: 固定 最大输入电压: 7.5V 输出电压: 3.3V 最大输出电流: 500mA RS3236-3.3YUTDN4 是一款低压差线性稳压器&#x…

kkfileview部署踩坑记录——kkfile部署启动失败、预览出错、乱码问题的处理

预览失败&#xff1a;报错 org.jodconverter.core.office.OfficeException: Could not store document: testdoc.pdf java.lang.NullPointerException: Could not open document: 1691539546735.docx 错误原因 由于kkfileView在linux上默认使用openOffice来实现转换 解决方…

leetcode 2316.统计无向图中无法互相到达点对数

思路&#xff1a;并查集 其实就是连通块的一个变形题目&#xff0c;一般的连通块题目要我们求的是连通个数&#xff0c;或者能不能到达&#xff0c;这里反过来问了。 首先&#xff0c;我们用dfs也是可以做到的&#xff0c;在dfs中统计每一个连通块的个数&#xff0c;然后用乘…

Spring原理分析--获取Environment资源对象

1.使用getEnvironment()获取环境信息 ApplicationContext接口继承了EnvironmentCapable接口&#xff0c;可以通过getEnvironment()获取Environment配置信息&#xff0c;例如&#xff1a; SpringBootApplication public class A01 {public static void main(String[] args) th…

解决方案:对数据进行负采样随机抽取1000W,用Pandas如何实现

文章目录 一、现象二、解决方案 一、现象 做建模的时候&#xff0c;有时候需要对数据进行负采样&#xff0c;就需要随机抽取数据&#xff0c;之前用SQL实现过order by rand()&#xff0c;附上链接解决方案&#xff1a;用户号出现多行&#xff0c;如何从中取其一并随机抽取100个…

PX4FMU和PX4IO最底层启动过程分析(下)

PX4FMU和PX4IO最底层启动过程分析&#xff08;下&#xff09; PX4FMU的系统启动函数为nash_main(int argc,char *argv[]) PX4IO的系统启动函数为nash_start(int argc,char *argv[]) PX4FMU启动函数nash_main(int argc,char *argv[]) 首先分析一下nash_main(int argc,char *a…

高效视频剪辑:视频批量调色,如何利用色调调整提升效率

在视频剪辑的后期处理中&#xff0c;调色是一个至关重要的环节。它不仅能够改变视频的整体氛围和风格&#xff0c;还能够突出视频的重点&#xff0c;增强观众的视觉体验。然而&#xff0c;对于大量的视频素材进行逐个调色处理&#xff0c;无疑会耗费大量的时间和精力。我们可以…

软件安装及YOLOv8环境配置及验证

先附上本章中所用到的软件及环境安装包&#xff0c;还有YOLOv8各任务权重&#xff1a; 软件及环境配置链接&#xff1a;https://pan.baidu.com/s/1-n2HJybicA6vW1YXfGRtcA 提取码&#xff1a;6vh8 YOLOv8各权重&#xff1a;链接&#xff1a;https://pan.baidu.com/s/1ApYUrJ_s…

C++相关概念和易错语法(12)(迭代器、string容量调整)

1.迭代器&#xff08;以string为例&#xff09; &#xff08;1&#xff09;基本理解&#xff1a;在我们刚接触迭代器的时候&#xff0c;我们可以将迭代器理解为改造过的“指针”&#xff0c;这是一个新的类型&#xff0c;指向对应容器中的各个元素。我们可以像指针那样对迭代器…

Lombok介绍、使用方法和安装

目录 1 Lombok背景介绍 2 Lombok使用方法 2.1 Data 2.2 Getter/Setter 2.3 NonNull 2.4 Cleanup 2.5 EqualsAndHashCode 2.6 ToString 2.7 NoArgsConstructor, RequiredArgsConstructor and AllArgsConstructor 3 Lombok工作原理分析 4. Lombok的优缺点 5. 总结 1 …