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

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

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

相关文章

SOLIDWORKS实用技巧——工程图模板替换

概述 工程师常在出图时选择最佳模板,在编辑一段时间后,发现需要更改图纸大小,怎样更改图纸大小还不影响现有工作。你是否也有此类问题? 那么,新建工程图时的模板从哪里来?如何轻松替换已有工程图的图纸格…

IDEA-SVN合并分支到主干

IDEA-SVN合并branch分支到主干master 1.选择VCS的 Integrate Project 2.选择分支合并 Source1 是合并后的分支 , 主分支 master Source2 是被合并的分支 , 分支 branch Try merge 可以尝试是否可以能够被合并,并且无冲突 3.合并完成后当前项目会出现需要提交的内容,检查一…

CSS特效014: hover后左右开门的效果

CSS常用示例100专栏目录 本专栏记录的是经常使用的CSS示例与技巧,主要包含CSS布局,CSS特效,CSS花边信息三部分内容。其中CSS布局主要是列出一些常用的CSS布局信息点,CSS特效主要是一些动画示例,CSS花边是描述了一些CSS…

汽车级芯片NCV7518MWATXG 可编程六沟道低压侧 MOSFET预驱动器 特点、参数及应用

NCV7518MWATXG 可编程六沟道低压侧 MOSFET 预驱动器属于 FLEXMOS™ 汽车级产品,用于驱动逻辑电平 MOSFET。该产品可通过串行 SPI 和并行输入组合控制。该器件提供 3.3 V/5 V 兼容输入,并且串行输出驱动器可以采用 3.3 V 或 5 V 供电。内部通电重置提供受…

【Linux 进入救援模式】

要在CentOS 7中进入单用户模式并修改IP地址,可以按照以下步骤进行操作: 在启动过程中按下键盘上的“e”键进入GRUB编辑器。 找到启动项中以“linux16”或“linuxefi”开头的行,并在该行末尾添加“单用户模式(single)…

InfoNCE Loss公式及源码理解

InfoNCE Loss公式及源码理解–从交叉熵损失谈起 当谈论到信息论中的损失函数时,InfoNCE(Noise Contrastive Estimation)和交叉熵损失都是两个关键的概念。它们不仅在衡量概率分布之间的差异方面发挥着重要作用,而且在深度学习的自…

做进销存什么软件好用

进销存软件是企业管理库存、采购和销售等环节的信息化管理系统,对于企业的运营和管理具有重要的意义。在选择进销存软件时,需要考虑以下因素: 功能需求:选择能够满足企业实际需求的进销存软件。例如,系统是否支持商品…

边缘计算系统设计与实践

💂 个人网站:【 海拥】【神级代码资源网站】【办公神器】🤟 基于Web端打造的:👉轻量化工具创作平台💅 想寻找共同学习交流的小伙伴,请点击【全栈技术交流群】 随着物联网、大数据和人工智能等技术的快速发展…

使用低代码可视化开发平台快速搭建应用

目录 一、JNPF可视化平台介绍 二、搭建JNPF可视化平台 【表单设计】 【报表设计】 【流程设计】 【代码生成器】 三、使用JNPF可视化平台 1.前后端分离: 2.多数据源: 3.预置功能: 4.私有化部署: 四、总结 可视化低代码…

【云原生】Spring Cloud Alibaba 之 Gateway 服务网关实战开发

目录 一、什么是网关 ⛅网关的实现原理 二、Gateway 与 Zuul 的区别? 三、Gateway 服务网关 快速入门 ⛄需求 ⏳项目搭建 ✅启动测试 四、Gateway 断言工厂 五、Gateway 过滤器 ⛽过滤器工厂 ♨️全局过滤器 六、源码地址 ⛵小结 一、什么是网关 Spri…

打包项目报错:程序包javax.servlet不存在

背景: WebService项目在没有配置Tomcat的情况下重新打包,由于是直接导入别人写好的项目,没有配置其他环境,所以报错程序包javax.servlet不存在 解决方法: 找到servlet-api.jar包,导入到现有项目的SDK 重…

Java,数据结构与集合源码,数据结构概述

目录 数据结构概念: 数据结构的研究对象: 研究对象一,数据间逻辑关系: 研究对象二,数据的存储结构(或物理结构): 研究对象三:运算结构 数据结构的相关介绍&#xff…

LeetCode [中等] 49. 字母异位词分组

给你一个字符串数组,请你将 字母异位词 组合在一起。可以按任意顺序返回结果列表。 字母异位词 是由重新排列源单词的所有字母得到的一个新单词。 思路: 利用字符数组的排序,将字符串数组中的每个字符串转换为字符数组,进行排序…

BGP的基础知识

BGP——边界网关协议 IGP——内部网关协议——OSPF、RIP、ISIS EGP——外部网关协议——EGP、BGP 边界网关协议BGP是一种实现自治系统AS之间的路由可达,并选择最佳路由的路径矢量路由协议。目前在IPV4环境下主要使用BGPV4,目前市场上也存在BGPV4&…

使用ExLlamaV2量化并运行EXL2模型

量化大型语言模型(llm)是减少这些模型大小和加快推理速度的最流行的方法。在这些技术中,GPTQ在gpu上提供了惊人的性能。与非量化模型相比,该方法使用的VRAM几乎减少了3倍,同时提供了相似的精度水平和更快的生成速度。 ExLlamaV2是一个旨在从…

SpringBoot : ch04 整合数据源

前言 Spring Boot 是当今最流行的 Java 开发框架之一,它以简洁、高效的特点帮助开发者快速构建稳健的应用程序。在实际项目中,涉及到数据库操作的需求时,我们需要对数据源进行整合。本文将重点介绍如何在 Spring Boot 中整合数据源&#xff…

NX二次开发UF_CAM_PREF_ask_integer_value 函数介绍

文章作者:里海 来源网站:https://blog.csdn.net/WangPaiFeiXingYuan UF_CAM_PREF_ask_integer_value Defined in: uf_cam_prefs.h int UF_CAM_PREF_ask_integer_value(UF_CAM_PREF_t pref, int * value ) overview 概述 This function provides the …

如何看待程序员领域内的“内卷”现象?

要搞清楚这个问题,我首先就来阐释一下“内卷”的概念。 内卷本身是从一个学术名词演化为网络流行词的,本是指文化模式因达到某种最终形态,既无法保持稳定也不能转化为更高级的新形态,而只能在这种文化模式内部无限变得复杂的现象。…

TVS瞬态抑制二极管的工作原理和特点?|深圳比创达电子EMC

TVS二极管一般是用来防止端口瞬间的电压冲击造成后级电路的损坏。防止端口瞬间的电压冲击造成后级电路的损坏。有单向与双向之分,单向TVS一般应用于直流供电电路,双向TVS应用于交流供电电路。 TVS产品的额定瞬态功率应大于电路中可能出现的最大瞬态浪涌…

【C++】const与类(const修饰函数的三种位置)

目录 const基本介绍 正文 前: 中: 后: 拷贝构造使用const 目录 const基本介绍 正文 前: 中: 后: 拷贝构造使用const const基本介绍 const 是 C 中的修饰符,用于声明常量或表示不可修改的对象、函数或成员函数。 我们已经了解了const基本用法,我们先进行…