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

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

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

一、Python 装饰器的简单介绍

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

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

@my_decorator
def my_function():
    pass

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

def my_function():
    pass

my_function = my_decorator(my_function)

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

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

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

import time


def log1(func):
    def wrapper1():
        start = time.time()
        res = func()
        print("exec time: %.2f" % (time.time()-start))
        return res

    print("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 res

    print("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 time


def log1(func):
    def wrapper1():
        start = time.time()
        res = func()
        print("exec time: %.2f" % (time.time()-start))
        return res

    print("log1 wrapper1")
    return wrapper1


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 res

    print("log2 wrapper2")
    return wrapper2


def 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 res

        return wrapper3

    print("log2 wrapper2")
    return wrapper2


@log2()
def func2():
    time.sleep(0.15)
    print("call func2")


if __name__ == '__main__':
    func2()
    # 等价于 func2 = log2()(func2) 这里它还是一个函数(可调用),而不是一个返回值了()

在这里插入图片描述

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

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

KEIL declaration may not appear after executable statement in block

KEIL declaration may not appear after executable statement in block 这个问题也是比较经典,就是不允许你的变量定义位置不允许在下边的代码区域,只允许在最上方 ‍ 修改编码模式为C99解决 ‍ ​​

机器学习周报第41周

目录 摘要Abstract一、文献阅读1.1 摘要1.2 背景1.3 论文方法1.3.1 局部特征提取1.3.2 局部特征转换器 (LoFTR) 模块1.3.4 建立粗粒度匹配1.3.5 精细匹配 1.4 损失1.5 实现细节1.6 实验1.6.1 单应性估计1.6.2 相对位姿估计 二、论文代码总结 摘要 本周阅读了一篇特征匹配领域的…

Git团队协作机制

Git 团队协作机制 1.团队内协作 小故事:岳不群手里有华山剑法但是不完整,需要弟子令狐冲进行完善,岳不群将华山剑法推送(push)到代码托管中心,这样岳不群就有属于自己的远程库,令狐冲从远程库…

思通数科大模型在智能数据查询系统中的深度应用:销售数据分析的革新

在企业决策支持系统中,销售数据分析占据着举足轻重的地位。思通数科的大模型技术,结合自然语言处理(NLP)和机器学习,为智能数据查询系统提供了强大的分析能力。本文将详细描述思通数科大模型在销售数据分析中的应用&am…

异常检测的学习和实战

1.应用: 1.在工业上的应用 当检测设备是否处于异常工作状态时,可以由上图分析得到:那些零散的点对应的数据是异常数据。因为设备大多数时候都是处于正常工作状态的,所以数据点应该比较密集地集中在一个范围内,而那些明…

Excel快速填充序号的方法

Excel快速填充序号常用的方法。 方法一:填充前面序号后拖拽 特点: 能有规律的填充,排序的行数由拖拽的行数决定。 此方法填充的序号等效于手打的序号,删除一行后下一行不会自动更新排序。 步骤:输入两个初始序号&…

Spring Cloud | “微服务“ 架构 与 Spring Cloud

“微服务” 架构 与 Spring Cloud 目录: "微服务" 架构 与 Spring Cloud1. 认识架构"单体" 架构"SOA" 架构"微服务" 架构 2. "微服务架构" 的功能 :① 微服务架构的 "自动化部署"② 服务 "集中化管理"③…

SpringBoot 使用logback(多环境配置)

Logback是由log4j创始人设计的又一个开源日志组件。可用于项目日志功能。官网地址 第1步&#xff1a;添加坐标依赖 <!--logback--> <dependency><groupId>ch.qos.logback</groupId><artifactId>logback-classic</artifactId><version…

ESP-01S刷固件ESP8266_NonOS_AT_Bin_V1.7.5_1 笔记240510

ESP-01S刷固件ESP8266_NonOS_AT_Bin_V1.7.5_1 笔记240510 固件下载地址 ESP-AT固件页面: https://www.espressif.com.cn/zh-hans/products/sdks/esp-at/resource 直接下载ESP8266 NonOS AT Bin V1.7.5.zip: https://www.espressif.com.cn/sites/default/files/ap/ESP8266_No…

Baidu Comate:释放编码潜能,革新软件开发

Baidu Comate Baidu Comate&#xff0c;智能代码助手&#xff0c;凭借着文心大模型的强大支撑&#xff0c;结合了百度多年的编程实战数据和丰富的开源资源&#xff0c;形成了一款崭新的编码辅助利器。它不仅具备着高智能、多场景、价值创造的特质&#xff0c;更可广泛应用于各…

sklearn之线性回归——以上证红利指数为例

文章目录 线性回归概念使用sklearn实现上证中立指数预测内置数据集的加载与处理 外部数据集的加载和处理数据内容数据加载和处理 开始预测分割数据集导入线性回归模型查看线性回归模型的系数绘制预测结果预测效果评估 最终代码 线性回归 线性回归&#xff08;Linear Regressio…

flutter开发实战-人脸识别相机使用

flutter开发实战-人脸识别相机使用 当需要拍摄的时候&#xff0c;需要检测到人脸再进行后续的操作&#xff0c;这里使用的是face_camera 一、引入face_camera 在工程的pubspec.yaml中引入插件 # 检测人脸face_camera: ^0.0.8iOS端需要设置相关权限 在info.plist文件中&…

数据库出现死锁的解决方法参考

死锁引起的原因一般是多个用户并发访问数据库导致的问题&#xff0c;或是因为某个进程挂死以后资源未释放导致的。通过onstat –p可查看deadlks项大于0即表示历史总计死锁次数。对于被锁的表进行操作的时候会出现-143 ISAM error: deadlock detected的错误。当其他会话访问此表…

YOLOv5 V7.0 - rknn模型的验证 输出精度(P)、召回率(R)、mAP50、mAP50-95

1.简介 RKNN官方没有提供YOLOv5模型的验证工具&#xff0c;而YOLOv5自带的验证工具只能验证pytorch、ONNX等常见格式的模型性能&#xff0c;无法运行rknn格式。考虑到YOLOv5模型转换为rknn会有一定的精度损失&#xff0c;但是需要具体数值才能进行评估&#xff0c;所以需要一个…

【目标检测论文解读复现NO.37】基于改进的 YOLOv8 变电设备红外图像检测

前言 此前出了目标改进算法专栏&#xff0c;但是对于应用于什么场景&#xff0c;需要什么改进方法对应与自己的应用场景有效果&#xff0c;并且多少改进点能发什么水平的文章&#xff0c;为解决大家的困惑&#xff0c;此系列文章旨在给大家解读最新目标检测算法论文&#xff0c…

Python 将Excel转换为多种图片格式(PNG, JPG, BMP, SVG)

目录 安装Python Excel库 使用Python将Excel工作表转换为PNG&#xff0c;JPG或BMP图片 使用Python将Excel特定单元格区域转换为PNG&#xff0c;JPG或BMP图片 使用Python将Excel工作表转换为SVG图片 有时&#xff0c;你可能希望以图片形式分享Excel数据&#xff0c;以防止他…

【C++】string底层的实现原理(简单详细)

前言 本篇文章我将按照C文档库中的模块顺序来实现和讲解其实现原理&#xff0c;我们只讲各板块中常用的 目录 一&#xff0c;Member functions&#xff08;成员函数&#xff09; 二、Iterators&#xff08;迭代器&#xff09; 三、Capacity&#xff08;容器&#xff09; 常…

ElasticSearch集群环境

ElasticSearch集群环境 1、Linux单机 下载地址&#xff1a;LINUX X86_64 (elastic.co) 下载之后进行解压 tar -zxf elasticsearch-7.8.0-linux-x86_64.tar.gz 名字太长了改个名字改成es mv elasticsearch-7.8.0 es因为安全问题&#xff0c;Elasticsearch 不允许 root 用户…

基于单片机的空气质量检测系统设计(51+4G版)-设计说明书

设计摘要&#xff1a; 本设计是基于单片机的空气质量检测系统设计涉及以下主要功能&#xff0c;旨在监测甲烷和一氧化碳的浓度&#xff0c;并在浓度过高时采取相应措施&#xff0c;以确保室内空气质量的安全。该系统使用传感器对甲烷和一氧化碳的浓度进行检测。传感器将收集到…

虚拟化技术 安装和配置StartWind iSCSI目标服务器

一、实验内容 安装StartWind iSCSI目标服务器配置StartWind iSCSI目标服务器 二、实验主要仪器设备及材料 安装有64位Windows操作系统的台式电脑或笔记本电脑&#xff0c;建议4C8G或以上配置已安装vSphere Client已创建虚拟机并在其上安装CentOS6.5StarWind安装介质starwind.…