Python 装饰器用法详解

目录

一、基本概念

二、语法形式

三、用法示例

1、用于日志记录

2、用于性能测试

3、用于事务处理

4、用于缓存结果

5、用于权限验证

总结


Python装饰器是Python中一种非常有用且强大的工具,它允许我们在不修改原有函数或类的基础上,对它们进行增强或修改。装饰器本质上是一个函数,它接受另一个函数作为参数,并返回一个新的函数。装饰器的语法形式为@decorator,它表示将一个函数进行装饰。

在Python中,装饰器广泛应用于各种场景,如日志记录、性能测试、事务处理等。通过使用装饰器,我们可以轻松地将一些通用的逻辑或行为附加到函数或类上,而无需在每个函数或类中重复编写相同的代码。

下面我们将详细介绍Python装饰器的基本概念、语法、用法和最佳实践,并通过丰富的代码示例进行演示。

一、基本概念

在Python中,装饰器是一个函数,它接受另一个函数作为参数,并返回一个新的函数。装饰器通常用于修改或增强原有函数的行为。通过使用装饰器,我们可以将一些通用的逻辑或行为(如日志记录、性能测试、事务处理等)附加到函数或类上,而无需在每个函数或类中重复编写相同的代码。

二、语法形式

装饰器的语法形式为@decorator,它表示将一个函数进行装饰。下面是一个简单的示例:

def my_decorator(func):  
    def wrapper():  
        print("Before the function is called.")  
        func()  
        print("After the function is called.")  
    return wrapper  
  
@my_decorator  
def say_hello():  
    print("Hello!")

在这个例子中,我们定义了一个名为my_decorator的装饰器函数,它接受一个函数作为参数,并返回一个新的函数wrapper。wrapper函数在调用原有函数之前和之后输出一些信息。然后我们使用@my_decorator将say_hello函数进行装饰。

当我们调用say_hello函数时,实际上会执行wrapper函数,它会在调用say_hello之前和之后输出信息。

三、用法示例

1、用于日志记录

装饰器可以用于日志记录,自动记录函数的执行时间和输出信息。下面是一个示例:

import time  
  
def log_execution_time(func):  
    def wrapper(*args, **kwargs):  
        start_time = time.time()  
        result = func(*args, **kwargs)  
        end_time = time.time()  
        print(f"Function {func.__name__} took {end_time - start_time:.6f} seconds to execute.")  
        return result  
    return wrapper  
  
@log_execution_time  
def some_function():  
    time.sleep(1)  
    print("Function executed.")

这个示例中,我们定义了一个名为log_execution_time的装饰器函数,它接受一个函数作为参数,并返回一个新的函数wrapper。wrapper函数在调用原有函数之前和之后记录时间,并输出函数的执行时间。然后我们使用@log_execution_time将some_function函数进行装饰。当我们调用some_function函数时,实际上会执行wrapper函数,它会在调用some_function之前和之后记录时间,并输出执行时间。

2、用于性能测试

装饰器还可以用于性能测试,自动测试函数的执行时间和内存使用情况。下面是一个示例:

import time  
import tracemalloc  
  
def measure_execution_time(func):  
    def wrapper(*args, **kwargs):  
        tracemalloc.start()  
        result = func(*args, **kwargs)  
        total_time = tracemalloc.stop() / 1000000.0  # convert to milliseconds  
        print(f"Function {func.__name__} took {total_time:.6f} ms to execute.")  
        return result  
    return wrapper

3、用于事务处理

装饰器还可以用于事务处理,自动管理函数的事务操作。下面是一个示例:

def transaction(func):  
    def wrapper(*args, **kwargs):  
        conn = database_connection()  # 获取数据库连接  
        try:  
            conn.begin()  # 开始事务  
            result = func(conn, *args, **kwargs)  
            conn.commit()  # 提交事务  
            return result  
        except:  
            conn.rollback()  # 回滚事务  
            raise  
        finally:  
            conn.close()  # 关闭数据库连接  
    return wrapper  
  
@transaction  
def some_function(conn, arg1, arg2):  
    # 在函数中执行数据库操作  
    # ...

这个示例中,我们定义了一个名为transaction的装饰器函数,它接受一个函数作为参数,并返回一个新的函数wrapper。wrapper函数在调用原有函数之前获取数据库连接,并开始事务。如果原有函数执行成功,则提交事务;如果发生异常,则回滚事务。最后,无论如何都会关闭数据库连接。然后我们使用@transaction将some_function函数进行装饰。当我们调用some_function函数时,实际上会执行wrapper函数,它会在调用some_function之前获取数据库连接,并管理事务。

4、用于缓存结果

装饰器还可以用于缓存函数的结果,避免重复计算。下面是一个示例:

def cache(func):  
    cache = {}  
    def wrapper(*args, **kwargs):  
        key = str(args) + str(kwargs)  
        if key in cache:  
            return cache[key]  
        else:  
            result = func(*args, **kwargs)  
            cache[key] = result  
            return result  
    return wrapper  
  
@cache  
def some_function(arg1, arg2):  
    # 在函数中执行耗时操作  
    # ...

这个示例中,我们定义了一个名为cache的装饰器函数,它接受一个函数作为参数,并返回一个新的函数wrapper。wrapper函数在调用原有函数之前检查缓存中是否已经存在结果,如果存在则直接返回结果;如果不存在则执行原有函数,并将结果存储在缓存中。然后我们使用@cache将some_function函数进行装饰。当我们多次调用some_function函数时,第一次执行原有函数并缓存结果;之后的调用直接返回缓存中的结果。这样可以避免重复计算,提高性能。

5、用于权限验证

装饰器还可以用于权限验证,自动验证函数调用是否符合权限要求。下面是一个示例:

def check_permission(func):  
    def wrapper(*args, **kwargs):  
        # 验证用户是否有权限调用该函数  
        if not user_has_permission():  
            raise PermissionDenied("You do not have permission to perform this action.")  
        return func(*args, **kwargs)  
    return wrapper  
  
@check_permission  
def some_function():  
    # 在函数中执行敏感操作  
    # ...


这个示例中,我们定义了一个名为check_permission的装饰器函数,它接受一个函数作为参数,并返回一个新的函数wrapper。wrapper函数在调用原有函数之前验证用户是否有权限调用该函数,如果没有权限则抛出PermissionDenied异常。

如果有权限则执行原有函数。然后我们使用@check_permission将some_function函数进行装饰。当我们调用some_function函数时,实际上会执行wrapper函数,它会在调用some_function之前验证用户权限。

总结

Python装饰器是一种非常灵活和强大的工具,可以用于各种场景,如日志记录、性能测试、事务处理、缓存结果和权限验证等。通过使用装饰器,我们可以轻松地将一些通用的逻辑或行为附加到函数或类上,而无需在每个函数或类中重复编写相同的代码。

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

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

相关文章

物联网AI MicroPython学习之语法 WDT看门狗外设

学物联网,来万物简单IoT物联网!! WDT 介绍 模块功能: 看门狗WDT(WatchDog Timer)外设驱动模块 接口说明 WDT - 构建WDT对象 函数原型:WDT(timeout)参数说明: 参数类型必选参数&#xff1f…

ultralytics yolov8 实例分割 训练自有数据集

参考: https://docs.ultralytics.com/datasets/segment/coco/ http://www.bryh.cn/a/613333.html 1、数据下载与转换yolo格式 1)数据集下载: 参考:https://universe.roboflow.com/naumov-igor-segmentation/car-segmetarion 下载的是coco格式,需要转换 2)coco2yolo t…

银行业务测试

1、商业银行四大类: 业务类系统、渠道类面试、MIS类系统、其他基础平台系统 2、银行系统开发流程(UAT是行方) 3、银行系统测试流程 4、对于不同的服务方式也不同,如:柜台、手机银行、网上银行,电话外呼,…

局域网共享打印机共享,简单至简至一键处理011bDll等问题

一、电脑系统是否激活(可选) 二、确保主客户端PC在同一局域网内(可选) 可以通过ping 目标地址 如ping 192.168.1.202;看是否可以正常通信 下面是惠普类型打印机共享问题关键(文本记得保存) …

MS2401隔离Σ-Δ调制器,可替代ADI的AD7401

产品简述 MS2401 是一款二阶 Σ-Δ 调制器,集成片上数字隔离器,能 将模拟输入信号转换为高速 1 位码流。调制器对输入信号连续 采样,无需外部采样保持电路。模拟信号输入满量程为 320 mV ,转换后的数字码流的最高数据速率为 2…

Unsupervised Condition GAN

Unsupervised Condition GAN主要有两种做法: Direct Transformation 直接输入domain X图片,经过Generator后生成对应的domain Y的图像。这种转化input和output不能够差太多。通常只能实现较小的转化,比如改变颜色等。 Projection to Commo…

低代码PaaS开发平台

目录 一、低代码概念 低代码目的 低代码核心功能 二、PaaS平台 PaaS服务的低代码平台 1.私有化部署,为数据安全保驾护航 2.业内领先技术,为开发强势赋能 3.超强集成能力,系统对接无忧 4.源代码交付,实现二开自由 三、小结 一、低代…

shell脚本三

目录 一、循环语句 一、循环 二、for循环语句 1.列表循环 2.与c语言循环相似的for循环 3.使用for打印三角形以及乘法表 4.测试172.16.114.0网段存活的主机并将存活的主机IP地址写入文件中,未存活的主机放入另一文件中 三、while循环语句 四、until循环语句…

银行数字化转型导师坚鹏:BLM银行数字化转型战略培训圆满结束

在数字化转型背景下,中国金融出版社金融文化研训院为了落实监管政策《关于银行业保险业数字化转型的指导意见》,充分认识到学习银行银行数字化转型战略的价值和重要性,特别举办《2023年金融机构数字化转型及数字化风控与运营管理研讨班》。为…

python 实现等声值线图绘制

今天讲一类环评项目的噪声预测 - 风电 风机噪声作为面源目前难有成熟的模型进行预测。根据国内外的研究,都是根据与风机中心的位置进行分级预测。 翟国庆等利用美国航天航空局(NASA)研发的风电机组噪声预测模型(以下简称 NASA”…

短时傅里叶变换函数编写

文章目录 傅里叶变换与短时傅里叶变换什么是窗?自己对手实现短时傅里叶变换 傅里叶变换与短时傅里叶变换 在了解短时傅里叶变换之前,首先要知道是什么是傅里叶变换( fourier transformation,FT),傅里叶变换…

吴恩达《机器学习》9-7-9-8:综合起来、自主驾驶

在神经网络的使用过程中,需要经历一系列步骤,从网络结构的选择到训练过程的实施。以下是使用神经网络时的主要步骤的小结: 一、网络结构的选择 输入层: 第一步是选择网络结构,即确定神经网络的层数以及每层的单元数。…

得物前端开发一面面经(等待结果中)

基本情况 上周有幸约到了得物的前端一面,问题都不是很难,但是比较底层,不是八股,而是js的很很细致的东西;且面试官会根据简历去问技术。本篇博客就记录一下这次一面面到的一些技术问题,以及我回答的情况。…

C++ DAY08 异常

概念 异常事件(如:除 0 溢出,数组下标越界,所要读取的文件不存在 , 空指针,内存不足 等等) 在 C 语言对错误的处理是两种方法: 一是使用整型的返回值标识错误; 二是使用 errn…

数字化转型:传统门店突破困境,实现可持续发展的必由之路

自2023年疫情管控基本解除以来,人民群众体验线下消费的意愿充分释放,夜经济、文娱文旅消费、暑期经济等线下消费场景持续走热。据统计,今年1-7月份,我国实体店零售额同比增长4.2%。虽然实体经济出现了消费复苏,发展向好…

【AT模式连接ONENET】ONENET可视化平台的使用

02 ONENET可视化平台的使用 ATCWMODE1 设置模式 ATCWDHCP1,1 启动DHCP功能 ①ATCWJAP"ssid","password" ATCWJAP“123456789”,“wang020118” ②ATMQTTUSERCFG0,1,"设备名字","设备ID","你的鉴权信息""…

气膜体育馆:低碳环保体育新潮流

在追求健康生活的今天,体育运动的重要性无法忽视。为了满足人民日益增长的体育需求,气膜体育馆应运而生,成为体育场馆领域的一次革命性创新。这种新型体育馆解决了传统体育场馆建设中面临的审批难、周期长、门槛高等问题,为我们的…

vue history路径编码

记录今天遇到的一个问题: 问题现状 有一个需要前端伪造302进行重定向的需求,我们需要将这样的一个路径:http://xxx.com/system-name/#/index,拼接在跳转地址的后面,进行重定向。拼接的方式是这样的: htt…

【JavaSE】-4-单层循环结构

回顾 运算符: 算术 --、逻辑 && & || |、比较 、三元 、赋值 int i 1; i; j i; //j2 i3 syso(--j"-----"i) //1 3 选择结构 if(){} if(){}else{} if(){}else if(){}else if(){}else{}//支持byte、short、int //支持char //支持枚举…

2018-2022年富时罗素 ESG评分数据

2018-2022年富时罗素 ESG评分数据 1、时间:2018-2022年 2、指标:证券代码、证券简称、富时罗素ESG评分、 3、说明: 富时罗素ESG评级体系评估了中国大陆、香港、欧洲以及美国等市场上1800家中国上市企业股票,评估了7200多种证券…