【Python 元编程】装饰器入门指南

news2025/1/10 23:45:00

在这里插入图片描述

Python装饰器入门指南🚀

在编程世界中,效率和优雅的代码往往是我们所追求的目标。Python 作为一种强大且灵活的编程语言,提供了一个称为“装饰器”的功能,让我们能够以一种简洁和优雅的方式扩展和管理我们的代码。

本文旨在为初学者提供一个关于 Python 装饰器的简明指南,帮助大家理解它们的基本概念、作用以及在实际编程中的应用。让我们开始这趟探索之旅,一步步揭开装饰器神秘的面纱吧!🌟

知识点📖

什么是装饰器? 🤔

官网:什么是装饰器?

以下内容引用于官网:

返回值为另一个函数的函数,通常使用 @wrapper 语法形式来进行函数变换。 装饰器的常见例子包括 classmethod()staticmethod()

装饰器语法只是一种语法糖,以下两个函数定义在语义上完全等价:

def f(arg):
    ...
f = staticmethod(f)

@staticmethod
def f(arg):
    ...

Python 中,装饰器是一种强大的编程结构,它们允许在不修改原有函数代码的情况下增强或改变函数的行为。装饰器本质上是一个函数,它接收一个函数作为参数并返回一个新的函数。

装饰器的作用 🚀

装饰器的主要作用是为现有的函数或方法添加额外的功能。它们提供了一种优雅的方式来扩展函数的功能,这在维护和调试代码时非常有用,特别是在遵循开放/封闭原则的情况下。

  • 修改函数或方法的行为:通过装饰器在不更改原始代码的情况下修改函数或方法的行为。例如,添加日志记录、性能计时器或输入验证。
  • 提高代码的可读性:装饰器允许将与函数相关的代码块独立封装,使代码更加清晰和易于理解。
  • 促进代码的重用:创建通用的装饰器,然后在多个函数或方法中重复使用它们,从而避免代码重复。

装饰器的应用场景 🌍

装饰器在许多应用场景中都非常有用,包括但不限于:

  • 日志记录:自动记录函数的调用细节。
  • 性能测试:检测函数运行时间。
  • 权限验证:检查调用者是否有权执行某个函数。
  • 缓存:为耗时的操作结果添加缓存。

为什么使用装饰器?💡

这里使用一个简单的案例来展开说明。

我有一份网络请求的代码如下所示,

import requests


def network_request():
    return requests.get(url='https://www.baidu.com').text[:10]


if __name__ == '__main__':
    network_request()

现在需要为它添加一个执行耗时的功能!

  • 啪的一下!很快啊,就添加了这个功能。这也太简单了!
import time


if __name__ == '__main__':
    st = time.time()
    res = network_request()
    print(res)
    et = time.time()
    print(f'共耗时{et - st} 秒')

紧接着,又需要请求其它网络请求,并且要分别计算它们的执行耗时。

啪的一下,还是很快,我改好了代码。简单的来又有点麻烦!

def network_request2():
    return requests.get(url='https://www.bilibili.com').text[:10]


if __name__ == '__main__':
    st = time.time()
    network_request()
    et = time.time()
    print(f'{network_request.__name__} 共耗时{et - st} 秒')
    #
    st = time.time()
    network_request2()
    et = time.time()
    print(f'{network_request2.__name__} 共耗时{et - st} 秒')

再紧接着,又有其它网络请求,并且要分别计算它们的执行耗时。

这时候我开始头疼了!于是我开始吃头疼药

于是我用上了 一个名叫 装饰器 功能!头竟然神奇的不疼了!!!

  • 在这里,我们定义了一个名为 timer 的装饰器,用于记录函数执行的时间。它是一个函数,接受一个函数作为传参,通过使用装饰器,我们可以轻松地为多个网络请求函数分别添加执行耗时的功能,而不必重复编写计时逻辑。这使得代码更加整洁和易于维护,同时提高了代码的可复用性。
def timer(func):
    def wrapper(*args, **kwargs):
        start_time = time.time()
        result = func(*args, **kwargs)
        end_time = time.time()
        print(f"{func.__name__} 执行时间: {end_time - start_time} 秒")
        return result

    return wrapper

这时再来看添加了执行耗时的代码,

  • 啪的一下,这次就很方便了!
@timer
def network_request():
    return requests.get(url='https://www.baidu.com').text[:10]
# 等同于 timer(network_request)

@timer
def network_request2():
    return requests.get(url='https://www.bilibili.com').text[:10]
# 等同于 timer(network_request2)


if __name__ == '__main__':
    print(network_request())
    print(network_request2())

看到这,你应该明白为什么需要使用以及什么时候需要装饰器了吧!

这个案例介绍的不够全面,但试想一下,你在接手别人的项目时候,想要添加一个日志记录或执行耗时的功能来扩展项目功能,你更想直接上手改代码,还是使用装饰器来完成呢!!!

代码实现

基础装饰器 🛠️

用回上述的代码,

import time

def timer(func):
    def wrapper(*args, **kwargs):
        start_time = time.time()
        result = func(*args, **kwargs)
        end_time = time.time()
        print(f"{func.__name__} 执行时间: {end_time - start_time} 秒")
        return result

    return wrapper

代码释义:

这个函数 timer 是一个装饰器函数,它的作用是为被装饰的函数添加计时功能。具体来说:

  1. 它接受一个函数 func 作为参数,这个函数即将被装饰的目标函数。

  2. timer 函数内部,定义了一个嵌套函数 wrapper,这个函数将替代原始的目标函数。

  3. wrapper 函数内部,记录了目标函数执行前的时间戳(start_time),然后调用原始的目标函数 func(*args, **kwargs) 来执行它,并记录执行后的时间戳(end_time)。

  4. 计算出函数执行的时间差,并使用 print 函数输出执行时间。

  5. 最后,wrapper 函数返回了原始函数 func 的执行结果,并成为了新的目标函数。

当你使用 @timer 装饰器来修饰一个函数时,它会自动将该函数替换为 wrapper 函数,从而在函数执行时会自动记录并输出执行时间,而无需修改原始函数的代码。这个装饰器可以用来统计函数的执行效率或者做性能分析。


接受传参的装饰器 📌

装饰器也可以接受外部参数,这需要在装饰器中添加另一个层级的函数:

import time
import requests


def repeat_decorator(repeat_time):
    def decorator(func):
        def wrapper(*args, **kwargs):
            for i in range(repeat_time):
                try:
                    return func(*args, **kwargs)
                except requests.RequestException as e:
                    print(f'第 {i} 次, 错误', e.args)

        return wrapper

    return decorator


@repeat_decorator(repeat_time=3)
def network_request():
    return requests.get(url='https://www.bilibili.com').text[:10]


if __name__ == '__main__':
    network_request()

代码释义:

这函数定义了一个装饰器 repeat_decorator,它的作用是用于处理函数执行时可能出现的异常并进行重试。具体介绍如下:

  1. repeat_decorator 是一个装饰器函数,它接受一个参数 repeat_time,表示重试的次数。

  2. repeat_decorator 内部,定义了一个嵌套的函数 decorator,这个函数接受一个函数 func 作为参数,这个 func 即将被装饰的目标函数。

  3. decorator 函数内部,定义了另一个嵌套函数 wrapper,这个函数将替代原始的目标函数。

  4. wrapper 函数内部,使用 for 循环进行多次尝试,尝试调用原始的目标函数 func(*args, **kwargs)

  5. 如果调用 func 过程中发生了 requests.RequestException 异常,它会捕获异常,并输出错误信息,同时继续进行下一次重试。

  6. 当函数成功执行或者达到了指定的重试次数后,返回最后一次执行的结果。

这个装饰器的作用是增强目标函数的健壮性,当目标函数可能因网络请求等原因而抛出异常时,它会尝试多次执行该函数,以增加函数的成功执行的机会。这对于处理不稳定的网络请求或者需要重试的操作非常有用。

输出:

  • 假设请求错误,则会打印以下结果,打印内容有删减!
0 次, 错误 (MaxRetryError("Failed to establish a new connection: [WinError 10061] 由于目标计算机积极拒绝,无法连接。')))"),)1 次, 错误 (MaxRetryError("Failed to establish a new connection: [WinError 10061] 由于目标计算机积极拒绝,无法连接。')))"),)2 次, 错误 (MaxRetryError("Failed to establish a new connection: [WinError 10061] 由于目标计算机积极拒绝,无法连接。')))"),)

为装饰器动态添加属性 ✨

装饰器还可以用来动态地给函数添加属性。以下是一个使用partial函数和setattr的示例:

  • 这个装饰器允许我们动态地将test2函数作为test的一个属性添加上去。
from functools import partial

def attach_wrapper(obj, func=None):
    if not func:
        return partial(attach_wrapper, obj)
    setattr(obj, func.__name__, func)
    return func

def test():
    print('This is test.')

@attach_wrapper(test)
def test2():
    print('This is test 2.')

if __name__ == '__main__':
    test()
    test.test2()

代码释义:

这添加属性的装饰器允许你将一个函数动态地添加为另一个函数的属性。具体作用如下:

  1. 定义了一个装饰器函数 attach_wrapper,它接受两个参数:objfuncobj 表示要附加属性的目标对象(通常是一个函数),而 func 是要添加为属性的函数。

  2. 如果 func 为空,则返回一个 partial 函数,该 partial 函数接受一个参数 obj,用于表示要附加属性的目标对象。这是一种延迟调用的方式,允许你稍后再次调用装饰器并传递 func 参数。

  3. 如果 func 不为空,它将使用 setattr 函数将 func 添加为 obj 的属性,属性名称为 func.__name__,即 func 函数的名称。

  4. 装饰器函数的返回值是 func,因此它不会影响原始函数的行为。

通过这种装饰器,你可以将一个函数(例如 test2)动态地添加为另一个函数(例如 test)的属性。这可以在一些情况下提供便利,使你能够轻松地访问和管理相关的函数,特别是在需要扩展函数的功能或者将一些相关的操作组织在一起时。

但事实上,上面的代码等价于下面:

def test():
    print('This is test.')

    
def test2():
    print('This is test 2.')


setattr(test, test2.__name__, test2)

总结

通过本文的学习,我们不仅掌握了 Python 装饰器的基础知识,还了解了它们在实际编程中的多种应用场景。装饰器不仅提高了代码的可重用性和可维护性,还增强了代码的可读性和优雅度。它们是 Python 编程中不可或缺的一部分,无论是简化代码、增加功能还是进行性能分析,都能发挥重要作用。🚀🚀🚀

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

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

相关文章

利用appium自动控制移动设备并提取数据

安装appium-python-client模块并启动已安装好的环境 安装appium-python-client模块 在window的虚拟环境下执行pip install appium-python-client 启动夜神模拟器,进入夜神模拟器所在的安装路径的bin目录下,进入cmd终端,使用adb命令建立adb…

生产环境 OpenFeign 的配置最佳实践

基础使用 OpenFeign 全方位讲解 1. 生产环境 OpenFeign 的配置事项 1.1 如何更改 OpenFeign 默认的负载均衡策略 warehouse-service: #服务提供者的微服务IDribbon:#设置对应的负载均衡类NFLoadBalancerRuleClassName: com.netflix.loadbalancer.RandomRule1.2 开启默认的 O…

C++ //练习 2.14 下面的程序合法吗?如果合法,它将输出什么?

C Primer&#xff08;第5版&#xff09; 练习 2.14 练习 2.14 下面的程序合法吗&#xff1f;如果合法&#xff0c;它将输出什么&#xff1f; int i 100, sum 0; for(int i 0; i ! 10; i)sum i; std::cout<<i<<" "<<sum<<std::endl;环境…

scipy通过快速傅里叶变换实现滤波

文章目录 fft模块简介fft函数示例滤波 fft模块简介 scipy官网宣称&#xff0c;fftpack模块将不再更新&#xff0c;或许不久之后将被废弃&#xff0c;也就是说fft将是唯一的傅里叶变换模块。 Fourier变换极其逆变换在数学上的定义如下 F ( ω ) ∫ − ∞ ∞ f ( t ) e − i ω…

Python图像处理【19】基于霍夫变换的目标检测

基于霍夫变换的目标检测 0. 前言1. 使用圆形霍夫变换统计图像中圆形对象2. 使用渐进概率霍夫变换检测直线2.1 渐进霍夫变换原理2.2 直线检测 3. 使用广义霍夫变换检测任意形状的对象3.1 广义霍夫变换原理3.2 检测自定义形状 小结系列链接 0. 前言 霍夫变换 (Hough Transform,…

2024最新:optee系统开发精讲 - 课程介绍

&#xff08;本课程中如有涉及代码或硬件架构&#xff0c;则对应的版本号&#xff1a;TF-A 2.80&#xff0c;optee 3.20, Linux Kernel 6.3&#xff0c;armv8.79.0的aarch64&#xff09; &#xff08;注意&#xff1a; 该课程没有PPT&#xff0c;该课程是对照代码讲解的&#x…

回归预测 | Matlab基于ABC-SVR人工蜂群算法优化支持向量机的数据多输入单输出回归预测

回归预测 | Matlab基于ABC-SVR人工蜂群算法优化支持向量机的数据多输入单输出回归预测 目录 回归预测 | Matlab基于ABC-SVR人工蜂群算法优化支持向量机的数据多输入单输出回归预测预测效果基本描述程序设计参考资料 预测效果 基本描述 1.Matlab基于ABC-SVR人工蜂群算法优化支持…

矩阵重叠问题判断

创作背景 看到一道题目有感而发想写一篇题解&#xff0c;涉及的是一种逆向思维 桌面窗体重叠 - 洛谷https://www.luogu.com.cn/problem/U399827题目来源于《信息学奥赛课课通》 大致就是给一个长方形的左上顶点坐标&#xff08;x1,y1&#xff09;和右下顶点坐标&#xff08;x…

面试题:SpringBoot项目怎么设计业务操作日志功能?

文章目录 前言需求描述与分析系统日志操作日志 设计思路Spring AOPFilter和HandlerInterceptorSpringAOP、过滤器、拦截器对比 实现方案环境配置依赖配置表结构设计代码实现 测试调试方法验证结果 总结 前言 很久以前都想写这篇文章&#xff0c;一直没有空&#xff0c;但直到现…

【QT+QGIS跨平台编译】之一:【sqlite+Qt跨平台编译】(一套代码、一套框架,跨平台编译)

文章目录 一、sqlite3介绍二、文件下载三、文件分析四、pro文件五、编译实践 一、sqlite3介绍 SQLite是一款轻型的数据库&#xff0c;是遵守ACID的关系型数据库管理系统&#xff0c;它包含在一个相对小的C库中。它是D.RichardHipp建立的公有领域项目。它的设计目标是嵌入式的&…

MSVS C# Matlab的混合编程系列2 - 构建一个复杂(含多个M文件)的动态库:

前言: 本节我们尝试将一个有很多函数和文件的Matlab算法文件集成到C#的项目里面。 本文缩语: MT = Matlab 问题提出: 1 我们有一个比较复杂的Matlab文件: 这个MATLAB的算法,写了很多的算法函数在其他的M文件里面,这样,前面博客的方法就不够用了。会报错: 解决办法如下…

[学习笔记]刘知远团队大模型技术与交叉应用L3-Transformer_and_PLMs

RNN存在信息瓶颈的问题。 注意力机制的核心就是在decoder的每一步&#xff0c;都把encoder的所有向量提供给decoder模型。 具体的例子 先获得encoder隐向量的一个注意力分数。 注意力机制的各种变体 一&#xff1a;直接点积 二&#xff1a;中间乘以一个矩阵 三&#xff1a;…

Opncv模板匹配 单模板匹配 多模板匹配

目录 问题引入 单模板匹配 ①模板匹配函数: ②查找最值和极值的坐标和值: 整体流程原理介绍 实例代码介绍: 多模板匹配 ①定义阈值 ②zip函数 整体流程原理介绍 实例代码: 问题引入 下面有请我们的陶大郎登场 这张图片是我们的陶大郎,我们接下来将利用陶大郎来介绍…

恒悦sunsite博客2023年总结及2024年展望

一、2023年总结 一年如一日的坚持做好一件事并不是容易的事情&#xff0c;但是只要我们坚持下去&#xff0c;乘风破浪会有时&#xff0c;直挂云帆济沧海。   2023年是意义非凡的一年&#xff0c;年初的时候自己定下了两个目标&#xff1a;第一个是完成博客专家认证&#xff1…

HarmonyOS鸿蒙应用开发 (一、环境搭建及第一个Hello World)

万事开头难。难在迈出第一步。心无旁骛&#xff0c;万事可破。没有人一开始就能想清楚&#xff0c;只有做起来&#xff0c;目标才会越来越清晰。--马克.扎克伯格 前言 2024年1月16日&#xff0c;华为目前开启已HarmonyOS NEXT开发者预览版Beta招募&#xff0c;报名周期为1月15…

做好销售人员激励的3个要诀

企业合并是企业发展的重要战略手段之一。许多成长中的企业在经过一段时间的积累后&#xff0c;为了获得快速成长&#xff0c;实现规模效应&#xff0c;通常会采用合并的手段实现目标&#xff0c;同时企业会制定新型政策规范企业管理。但是在制定政策之前&#xff0c;企业通常会…

到店商详架构变迁

一、项目背景 到店商详是平台为京东到店业务提供的专属商详页面&#xff0c;将传统电商购物路径打造成以LBS门店属性的本地生活服务交易链路。 二、架构变迁 1、 主站商详扩展点 **优点&#xff1a;**到店侧仅关注业务&#xff0c;无需过度关注服务部署、性能优化等。 **缺…

Java实现大学计算机课程管理平台 JAVA+Vue+SpringBoot+MySQL

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 实验课程档案模块2.2 实验资源模块2.3 学生实验模块 三、系统设计3.1 用例设计3.2 数据库设计3.2.1 实验课程档案表3.2.2 实验资源表3.2.3 学生实验表 四、系统展示五、核心代码5.1 一键生成实验5.2 提交实验5.3 批阅实…

数字IC笔试题——门控时钟与控制信号电平、与门门控、或门门控、上升沿门控、下降沿门控

门控时钟问题。 &#xff08;华为-2019-芯片-数字-34&#xff09; 从后端设计考虑&#xff0c;在必须使用门控时钟的时候&#xff0c;需要遵循一个原则&#xff1a;门控时钟的输出只能跟着时钟信号进行跳变&#xff0c;而不能跟着控制信号进行跳变&#xff0c;也就是说对于用N…

【订单领域】如果订单要分库分表,如何确认最佳库表数量?

&#x1f389;欢迎来系统设计专栏&#xff1a;如果订单要分库分表&#xff0c;如何确认最佳库表数量? &#x1f4dc;其他专栏&#xff1a;java面试 数据结构 源码解读 故障分析 &#x1f3ac;作者简介&#xff1a;大家好&#xff0c;我是小徐&#x1f947;☁️博客首页&#x…