强大而灵活的python装饰器

news2024/9/21 10:39:00

装饰器(Decorators)

一、概述

在Python中,装饰器是一种特殊类型的函数,它允许我们修改或增强其他函数的功能,而无需修改其源代码。装饰器在函数定义之后立即调用,并以函数对象作为参数。装饰器返回一个新的函数对象,这个新函数对象通常会“包装”或“装饰”原函数,并在调用时执行额外的操作。

二、基本用法

装饰器的基本语法是使用@符号,后面紧跟装饰器函数的名称。装饰器函数通常接收一个函数作为参数,并返回一个新的函数。

@decorator_function  
def my_function():  
    pass

在上面的代码中,decorator_function是一个装饰器,它接收my_function作为参数,并返回一个新的函数对象。这个新的函数对象在调用时会执行decorator_function中定义的代码,并在适当的时候调用my_function

三、装饰器函数

装饰器函数通常接收一个函数作为参数,并返回一个新的函数。装饰器函数内部的新函数通常会调用原函数,并在调用前后执行额外的操作。

def decorator_function(func):  
    def wrapper(*args, **kwargs):  
        # 在原函数执行前执行的代码  
        print("Before function call")  
          
        # 调用原函数  
        result = func(*args, **kwargs)  
          
        # 在原函数执行后执行的代码  
        print("After function call")  
          
        return result  
      
    return wrapper

四、使用装饰器

使用装饰器非常简单,只需在函数定义之前加上@符号和装饰器函数的名称即可。

@decorator_function  
def my_function():  
    print("Inside function")  
  
# 调用被装饰的函数  
my_function()

五、使用装饰器的测试示例

下面是一个使用装饰器来实现基准测试的示例:

import time  
from functools import wraps  
  
def benchmark(func):  
    """  
    基准测试装饰器,用于测量函数的执行时间。  
  
    参数:  
        func (callable): 要测试的函数。  
  
    返回:  
        callable: 装饰后的函数,它会打印原始函数的执行时间。  
    """  
    @wraps(func)  
    def wrapper(*args, **kwargs):  
        start_time = time.time()  
        result = func(*args, **kwargs)  
        end_time = time.time()  
          
        elapsed_time = end_time - start_time  
        print(f"执行 {func.__name__} 花费了 {elapsed_time:.6f} 秒")  
        return result  
      
    return wrapper  
  
# 示例函数,我们将对其应用基准测试装饰器  
@benchmark  
def example_function(n):  
    sum = 0  
    for i in range(n):  
        sum += i  
    return sum  
  
# 调用示例函数,装饰器会自动打印执行时间  
print(example_function(1000000))

在这个例子中,benchmark 是一个装饰器函数,它接受一个函数作为参数,并返回一个新的函数 wrapperwrapper 函数会记录开始和结束时间,然后调用原始函数并打印出执行时间。@wraps(func) 是一个内置装饰器,用于保留原始函数的元信息(如函数名、文档字符串等)。

通过在 example_function 定义之前使用 @benchmark 装饰器,我们告诉Python在调用 example_function 时实际上要调用的是 wrapper 函数,而 wrapper 函数会记录并打印出 example_function 的执行时间。

使用装饰器的好处是你可以在不修改原始函数的情况下轻松添加新功能,例如基准测试。这使得代码更加模块化且易于维护。

六、装饰器的高级用法

1、多个装饰器

一个函数可以同时使用多个装饰器,只需在函数定义前依次列出它们即可。

@decorator1  
@decorator2  
def my_function():  
    pass

2、带参数的装饰器

装饰器本身也可以接收参数,并在内部定义装饰器函数。

def outer_decorator(param):  
    def decorator_function(func):  
        def wrapper(*args, **kwargs):  
            print(f"Before function call with param: {param}")  
            result = func(*args, **kwargs)  
            print(f"After function call with param: {param}")  
            return result  
          
        return wrapper  
      
    return decorator_function  
  
@outer_decorator("hello")  
def my_function():  
    print("Inside function")  
  
my_function()

七、注意事项

  • 装饰器会改变函数的名称和文档字符串,除非使用@wraps装饰器来保留这些信息。
  • 装饰器会改变函数的签名,这可能会影响依赖于函数签名的工具或库。
  • 装饰器可能会增加函数的调用开销,因为每次调用被装饰的函数时,都会执行额外的代码。

八、总结

装饰器是Python中一种强大而灵活的工具,它允许我们在不修改函数源代码的情况下,为函数添加额外的功能。通过正确使用装饰器,我们可以提高代码的可读性、可维护性和可扩展性。

文心大模型3.5生成

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

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

相关文章

【Qt】Sqlite数据库加密

1. 加密方式 对数据库文件加密。既不会暴露表结构,也不会暴露数据细节。 2. 加密工具(QtCipherSqlitePlugin) 用于密码 SQLite 的 Qt 插件,它基于 SQLite 源和 wxWidget 中的 wxSQLite3插件github地址:https://gith…

Vue ElementUI 修改消息提示框样式—messageBox 的大小

在窄屏模式下(移动端或pda),提示框的宽度太宽,会出现显示不完全的问题。 应当如何修改 ElementUI 的样式呢? open() {this.$confirm(window.vm.$i18n.t("tips.conLogOut"),window.vm.$i18n.t("tips.tip…

JVM内存回收算法

1.1 引用计数法 每个对象创建的时候,会分配一个引用计数器,当这个对象被引用的时候计数器就加1,当不被引用或者引用失效的时候计数器就会减1。任何时候,对象的引用计数器值为0就说明这个对象不被使用了,就认为是“垃圾…

回溯 Leetcode 47 全排列II

全排列II 给定一个可包含重复数字的序列 nums ,按任意顺序 返回所有不重复的全排列。 Leetcode 47 学习记录自代码随想录 示例 1: 输入:nums [1,1,2] 输出: [[1,1,2], [1,2,1], [2,1,1]] 示例 2: 输入&#xff1…

6. Z 字形变换

将一个给定字符串 s 根据给定的行数 numRows ,以从上往下、从左到右进行 Z 字形排列。 比如输入字符串为 "PAYPALISHIRING" 行数为 3 时,排列如下: P A H N A P L S I I G Y I R 之后,你的输出需要从左往右…

【MySQL】数据库中常用的函数

目录 聚合函数COUNT()函数的多种用法COUNT(*)COUNT(主键)COUNT(1)COUNT(常量)COUNT(非主键)COUNT(distinct(字段)) COUNT()函数小结 字符函数length(str)函数:获取参数值的字节个数concat(str1,str2,...)函数:字符串拼接upper(str)、lower(str)函数:大小…

雨云:为你拨开云雾见青天

一、雨云品牌概览 雨云,这名字一听就让人联想到蓝天白云,清爽自然。那么,这个品牌是否真的如其名,能为我们这些在数字世界中漂泊的旅人提供一片宁静、稳定的“云”呢?接下来,让我们深入了解雨云的资质、能…

Python学习 day06(类、对象、构造方法、私有方法、继承

类 程序中数据的组织多种多样,如果我们简单用变量来记录,就会混乱、不统一,如下所示: 类比现实中的表格,我们可以用类来组织数据,如下: 类的定义和使用 类中的变量叫做成员变量,类中…

逆序字符串

逆序字符串 题目描述:解法思路:解法代码:运行结果: 题目描述: 输入⼀个字符串,写⼀个函数将⼀个字符串的内容逆序过来。 测试1: 输⼊:abcdef 输出:fedcba 测试2&#x…

Docsify部署IIS

什么是Docsify? 一个神奇的文档网站生成器。docsify 可以快速帮你生成文档网站。不同于 GitBook、Hexo 的地方是它不会生成静态的 .html 文件,所有转换工作都是在运行时。如果你想要开始使用它,只需要创建一个 index.html 就可以开始编写文档…

Unity游戏项目中的优化之摄像机视锥体剔除优化

在项目中一个完成的游戏场景一般都会有成千上百的物体,假如都去让GPU全部渲染一遍,那带来的消耗其实是挺大的,很多不在摄像机范围内的物体其实没有必要去渲染,尽管GPU自带剔除,但是如果从CPU阶段就提交给GPU指令——哪…

Springboot项目中定时任务的四种实现方式

文章目录 1. 使用Scheduled注解1.1 时间间隔执行1.2 固定时间点执行 2. 使用EnableScheduling注解启用定时任务3. 实现SchedulingConfigurer接口4. 使用Quartz框架4.1 配置QuartzScheduler4.2 定义Job类和Trigger类 5. 总结 在开发现代应用时,定时任务是一个非常常见…

面试问答总结之并发编程

文章目录 🐒个人主页🏅JavaEE系列专栏📖前言:🎀多线程的优点、缺点🐕并发编程的核心问题 :不可见性、乱序性、非原子性🪀不可见性🪀乱序性🪀非原子性&#x1…

spring boot3解决跨域的几种方式

⛰️个人主页: 蒾酒 🔥系列专栏:《spring boot实战》 🌊山高路远,行路漫漫,终有归途。 目录 1.前言 2.何为跨域 3.跨域问题出现特征 4.方式一:使用 CrossOrigin 注解 5.方式二:自定义…

百度交出2023年业绩答卷:全力提速AI布局,注入业绩增长新动能

2月28日,百度集团(HK:09888、NASDAQ:BIDU,下称“百度”)发布2023年第四季度及全年财报,交出了一份营收与利润双双跃升的答卷,展现出百度在巩固原有业绩护城河的基础上,投入AI大模型后释放出的巨…

零拷贝技术深入分析

一、零拷贝 在前面的文章“深浅拷贝、COW及零拷贝”中对零拷贝进行过分析,但没有举例子,也没有深入进行展开分析。本文将结合实际的例程对零拷贝进行更深入的分析和说明。 在传统的IO操作中,以文件通过网络传输为例 ,一般会经历以…

【星海随笔】存储硬盘基础信息科普

市场上的磁盘分类有:IDE磁盘(多用于PC机)、SATA磁盘、SAS磁盘、SSD磁盘等 IDE 易于使用与价格低廉,问世后成为最为普及的磁盘接口。 速度慢、速度慢、速度慢。 ATA-7是ATA接口的最后一个版本,也叫ATA133。ATA133接口支…

【C++从0到王者】第四十六站:图的深度优先与广度优先

文章目录 一、图的遍历二、广度优先遍历1.思想2.算法实现3.六度好友 三、深度优先遍历1.思想2.代码实现 四、其他问题 一、图的遍历 对于图而言,我们的遍历一般是遍历顶点,而不是边,因为边的遍历是比较简单的,就是邻接矩阵或者邻接…

ChatGPT学习第四周

📖 学习目标 ChatGPT实践操作 通过实际操作和练习,加深对ChatGPT功能的理解。 项目:创建一个ChatGPT应用案例 设计一个基于ChatGPT的小项目,将理论应用于实践。 ✍️ 学习活动 学习资料 《万字干货!ChatGPT 从零完…

地图可视化绘制 | R-ggplot2 NC地图文件可视化

在推出两期数据分享之后,获取数据的小伙伴们也知道,数据格式都是NetCDF(nc) 格式网格数据,虽然我在推文分享中说明使用Python、R或者GIS类软件都是可以进行 处理和可视化绘制的,但是,还是有小伙伴咨询使用编程软件Pyth…