深入理解Python装饰器:丰富函数功能的强大工具

导语:装饰器是Python非常强大的功能之一,它们允许程序员修改或增强已有函数或方法的行为,而无需更改其本身的代码,这篇文章,让我们一起来看看(我自己这里理解的也不是很透彻)

       ----------更正---------

写完这篇文章,敲了不少示例,我感觉我透彻了

目录

装饰器基础

装饰器简介

创建简单的装饰器

示例:一个简单的装饰器

装饰器的工作原理

装饰器的使用场景

装饰器进阶

带参数的装饰器

示例:带参数的装饰器

示例:通用装饰器

装饰器的嵌套

示例:装饰器嵌套

装饰器的实际应用

日志记录装饰器

示例:日志记录装饰器

性能测试装饰器

示例:性能测试装饰器

权限校验和认证

示例:权限校验装饰器

注意事项

tips

注意!


装饰器基础

装饰器简介

        在Python中,装饰器是一种特殊的函数,它允许修改或增强其他函数或方法的功能,而不需要更改其自身代码。装饰器可以看作是对函数进行包装的包装器,它们在不修改原始函数的情况下,提供了一种灵活的方式来添加新功能。

创建简单的装饰器

        装饰器本身是一个函数,它接收一个函数作为参数,并返回一个新的函数。使用装饰器的常见方式是使用@语法,称为语法糖。

示例:一个简单的装饰器

def simple_decorator(func):def wrapper():print("函数前========")func()print("函数后========")return wrapper@simple_decorator
def say_hello():print("Hello world!")say_hello()

输出结果:

        在这个例子中,simple_decorator是一个装饰器,它在被装饰的函数say_hello执行前后添加了额外的打印语句。

装饰器的工作原理

当你使用@simple_decorator时,Python会执行以下步骤:

  1. say_hello函数作为参数传递给simple_decorator函数。
  2. simple_decorator函数返回wrapper函数。
  3. say_hello现在指向wrapper函数。

当我调用say_hello()时,实际上是在调用wrapper()函数。

装饰器的使用场景

  • 日志记录:自动记录函数的调用细节。
  • 性能测试:测量和报告函数的执行时间。
  • 访问控制和认证:在函数执行前进行权限校验。
  • 缓存和备忘:存储函数的结果以避免重复计算。

装饰器进阶

带参数的装饰器

        在某些情况下,你可能需要让装饰器接受额外的参数。这可以通过在装饰器外层再添加一个函数来实现。

示例:带参数的装饰器

def repeat(num_times):def decorator_repeat(func):def wrapper(*args, **kwargs):for _ in range(num_times):value = func(*args, **kwargs)return valuereturn wrapperreturn decorator_repeat@repeat(num_times=3)
def greet(name):print(f"Hello {name}")greet("World")

输出结果:

        在这个示例中,repeat是一个接受参数的装饰器工厂函数,它返回一个装饰器。这个装饰器随后应用于greet函数。

注意 !

        为了使装饰器更通用,能够适用于任意参数的函数,通常会在内部函数wrapper中使用*args**kwargs

示例:通用装饰器

def debug(func):def wrapper(*args, **kwargs):args_repr = [repr(a) for a in args]kwargs_repr = [f"{k}={v!r}" for k, v in kwargs.items()]signature = ", ".join(args_repr + kwargs_repr)print(f"===》 {func.__name__}({signature})")value = func(*args, **kwargs)print(f"{func.__name__!r} 返回的是: {value!r}")return valuereturn wrapper@debug
def make_greeting(name, age=None):greeting = f"{name}!"if age:greeting += f" 你今年 {age} 岁了"return greetingprint(make_greeting("小明", age=25))

返回结果:

在这个例子中,debug装饰器打印被装饰函数的调用细节,包括函数名称、传递的参数和返回值。

装饰器的嵌套

        装饰器可以嵌套使用,这意味着你可以在一个函数上应用多个装饰器,从而将多个功能叠加到该函数上。

示例:装饰器嵌套
#上面省略@debug
@repeat(num_times=2)
def say_hello(name):greeting = f"Hello {name}"print(greeting)return greetingsay_hello("Alice")

        在这个例子中,say_hello函数同时被debugrepeat装饰器装饰。装饰器的应用顺序是从最接近函数定义的装饰器开始向外应用

装饰器的实际应用

        装饰器不仅是一个理论概念,它们在实际编程中有着广泛的应用。通过具体的例子,我们可以更好地理解装饰器如何在各种情况下提供优雅的解决方案。

日志记录装饰器

        在应用程序中,跟踪函数的调用细节对于调试和性能分析非常重要。装饰器可以用来自动记录函数的调用和返回信息。

示例:日志记录装饰器

在这个例子中,log装饰器记录了函数名、参数和返回值。

import loggingdef log(func):def wrapper(*args, **kwargs):logging.info(f"Running {func.__name__} with arguments {args} and {kwargs}")output = func(*args, **kwargs)logging.info(f"{func.__name__} returned {output}")return outputreturn wrapper@log
def add(x, y):return x + y

性能测试装饰器

测量函数的执行时间是性能分析中的常见需求。装饰器可以被用来自动化这个过程。

示例:性能测试装饰器

在这个例子中,timeit装饰器测量并打印了函数的执行时间。

import timedef timeit(func):def wrapper(*args, **kwargs):start_time = time.time()result = func(*args, **kwargs)end_time = time.time()print(f"{func.__name__} took {end_time - start_time} seconds to run.")return resultreturn wrapper@timeit
def do_some_heavy_lifting():time.sleep(2)return "Task completed"

权限校验和认证

装饰器可以用于在函数执行前进行权限检查,这在Web开发和API设计中尤其有用。

示例:权限校验装饰器

在这个例子中,authorized装饰器检查用户是否具有执行特定函数的权限。

def authorized(roles):def decorator(func):def wrapper(*args, **kwargs):if user.role not in roles:raise Exception("Unauthorized")return func(*args, **kwargs)return wrapperreturn decorator@authorized(["admin", "user"])
def update_database():print("Database updated")

注意事项

tips

  • 使用functools.wraps装饰器来保留原函数的元数据(如名称和文档字符串)。
  • 虽然装饰器是一个强大的工具,但过度使用它们会使代码变得难以理解和维护。
  • 尽可能编写通用的装饰器,以便在不同的函数和项目中重用。
  • 给装饰器取一个清晰、描述性的名称,让其他开发者一眼就能理解装饰器的功能。

注意!

  1. 改变函数签名:使用*args**kwargs来接受任意数量的参数,这样可以确保装饰器不会因为函数参数更改而失败。

  2. 深层嵌套装饰器的复杂性:嵌套太多层装饰器可能会使代码变得复杂难懂。在可能的情况下,尝试将多个功能合并到一个装饰器中,或重构代码以减少嵌套。

  3. 装饰器与函数绑定:记住装饰器在定义时就绑定到其装饰的函数上,这意味着它们不能轻易地被重新应用于其他函数或方法。

  4. 状态管理:如果装饰器维护状态(比如计数),需要确保状态管理的线程安全和一致性。

--------------------------------

以上,欢迎评论交流、点个赞再走吧~

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

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

相关文章

ansible远程操作主机功能和自动化运维

ansible 两个功能:1、远程操作主机功能 2、自动化运维(play 剧本 yaml) 简述: 是基于python开发的配置管理和应用部署工具。在自动化运维中,现在是异军突起。 Asible能批量配置,部署,管理上千…

【从零开始学习JAVA集合 | 第一篇】深入解读HashMap源码(含面试题)

目录 目录 前言: HashMap简介: HashMap的常用常量和变量: HashMap的重要考点: HashMap的存储过程: HashMap的扩容过程: HashMap的初始化: 常见面试题: 总结:…

AI摄影绘画与PS优化:重塑数字艺术的未来

文章目录 《AI摄影绘画与PS优化从入门到精通》内容简介作者简介楚天 目录前言/序言 在科技日新月异的今天,人工智能(AI)已经渗透到我们生活的各个领域,包括艺术创作。AI摄影绘画和Photoshop(PS)优化是这个领…

「Verilog学习笔记」交通灯

专栏前言 本专栏的内容主要是记录本人学习Verilog过程中的一些知识点,刷题网站用的是牛客网 timescale 1ns/1nsmodule triffic_light(input rst_n, //异位复位信号,低电平有效input clk, //时钟信号input pass_request,output wire[7:0]clock,output reg…

使用Alpha Vantage API和Python进行金融数据分析

Alpha Vantage通过一套强大且开发者友好的数据API和电子表格,提供实时和历史的金融市场数据。从传统资产类别(例如股票、ETF、共同基金)到经济指标,从外汇汇率到大宗商品,从基本数据到技术指标,Alpha Vanta…

扫雪铲冰,绿萝护航平安出行

为保护行人出行安全,确保道路干净整洁,绿萝党员志愿者带头行动,号召志愿者积极参与扫雪铲冰。2023年12月16日,绿萝志愿服务队与拱辰集体林场携手在昊天碳中和公园开展了志愿扫雪活动。 行动是最好的融雪剂。大雪过后的道路是最为寒…

自动驾驶TPM技术杂谈 ———— 交通事件分类与编码

文章目录 介绍交通事件分类交通事件分类顺序码交通事故交通灾害交通气象路面状况道路施工活动重大事件其他 介绍 1. 交通事件 —— 由于人、车辆、设施、环境之间的不协调导致正常交通秩序的突发性混乱事件。  2. 交通管制 —— 公安机关交通管理部门根据法律、法规&#xff0…

2.1 数据操作

声明: 注意!!! 本博客仅用于本人学习笔记作用,所有资料来源都来自于李沐大神,博客中图片为沐神书上的图片。如有侵权,请联系本人删除,谢谢。 资料来源:https://zh-v2.d2…

什么是通讯芯片?通讯芯片又分什么种类?

通讯芯片是指在通讯系统中起到连接和传输信号的重要组成部分。根据通讯芯片的功能和应用,可以将其分为以下几类: 1、收发器芯片 2、窄带中频放大器芯片 3、卫星接收器LNB芯片 4、电话机芯片 5、压缩扩展器芯片 以上是通讯芯片的一些分类,随着通讯技术的不断发展…

patchless amsi学习(中)

DR7 DR7被称为“调试控制寄存器”,允许对每个硬件断点进行精细控制。其中,前8位控制是否启用了特定的硬件断点。偶数位(0、2、4、6)称为L0-L3,在本地启用了断点,这意味着仅在当前任务中检测到断点异常时才…

接口自动化测试实战教程

实现思路 使用excel管理用例用例信息,requests模块发送http请求,实现了记录日志,邮件发送测试报告的功能 目录结构如下: 下面直接上代码: 统筹脚本 # -*- coding:utf-8 -*-import osfrom interface import Interfa…

Android Termux安装MySQL数据库并通过内网穿透实现公网远程访问

文章目录 前言1.安装MariaDB2.安装cpolar内网穿透工具3. 创建安全隧道映射mysql4. 公网远程连接5. 固定远程连接地址 前言 Android作为移动设备,尽管最初并非设计为服务器,但是随着技术的进步我们可以将Android配置为生产力工具,变成一个随身…

【MongoDB】--MongoDB的Sort排序问题

目录 一、问题背景描述1.1、问题背景1.2、问题分析 二、建立索引支持深度翻页查询2.1、调整sort排序的内存限制【不建议】2.2、创建索引2.3、拓展--组合索引什么时候失效 二、聚合查询解决深度翻页查询 一、问题背景描述 1.1、问题背景 现实系统页面翻页到20000页之后&#x…

Improving IP Geolocation with Target-Centric IP Graph (Student Abstract)

ABSTRACT 准确的IP地理定位对于位置感知的应用程序是必不可少的。虽然基于以路由器为中心(router-centric )的IP图的最新进展被认为是前沿的,但一个挑战仍然存在:稀疏IP图的流行(14.24%,少于10个节点,9.73%孤立)限制了图的学习。为了缓解这个问题,我们将目标主机(ta…

【LeetCode刷题笔记(8-2)】【Python】【接雨水】【单调栈】【困难】

文章目录 引言接雨水题目描述提示 解决方案2:【单调栈】结束语 【接雨水】 【LeetCode刷题笔记(8-1)】【Python】【接雨水】【动态规划】【困难】 引言 编写通过所有测试案例的代码并不简单,通常需要深思熟虑和理性分析。虽然这…

200kw直流电子负载测试

在进行200kw直流电子负载测试时,需要设置合适的负载参数,如电流、电压和功率等,并记录测试结果。在测试过程中,可以通过改变负载参数来模拟不同工作条件下的电子设备的性能。通过对电子设备在不同负载条件下的响应进行测试和分析&…

JupyterHub 如何切换 conda 小环境

JupyterHub 如何切换 conda 小环境 服务器已经部署好 JupyterHub ,相关端口请看对应答疑群群公告。在Jupyterhub 中使用 conda 创建的小环境,首先 ssh 登录上服务器或者在 JupyterHub 网页端打开终端 terminal。然后安装 conda ,方法请见 Q4&…

【MyBatis Plus】逻辑删除、分页、乐观锁的应用及讲解

🎉🎉欢迎来到我的CSDN主页!🎉🎉 🏅我是Java方文山,一个在CSDN分享笔记的博主。📚📚 🌟推荐给大家我的专栏《MyBatis-Plus》。🎯🎯 &am…

改进灰狼算法求解:考虑需求响应的风-光柴-储容量优化配置

目录 文章摘要: 亮点: 研究背景: 考虑需求相应的容量配置: 风、光、柴、储微电网模型: 储能配置模型: 改进的灰狼算法: 基于余弦规律变化的收敛因子 引入动态权重策略 运行效果&#…

数据结构:图文详解 队列 | 循环队列 的各种操作(出队,入队,获取队列元素,判断队列状态)

目录 队列的概念 队列的数据结构 队列的实现 入队 出队 获取队头元素 获取队列长度 循环队列的概念 循环队列的数据结构 循环队列的实现 判断队列是否为空 判断队列是否已满 入队 出队 得到队头元素 得到队尾元素 队列的概念 队列(Queue&#xff0…