Day27 函数专题2:装饰器

1.装饰器的思想:进一步复用

装饰器(Decorator)是 Python 中一种强大的编程工具,核心作用是在不修改原函数代码的前提下,为函数添加额外功能(如日志记录、性能统计、权限校验等)。它充分利用了 Python 中 “函数是一等公民”(可作为参数传递、赋值、返回)的特性,通过 “包装” 原函数来扩展其行为。

装饰器本质是一个返回函数的高阶函数。它接收被装饰的函数作为参数,返回一个 “包装后的新函数”。这个新函数在保留原函数功能的基础上,增加了额外逻辑。

如果让一个函数具备太多功能,那么他看起来就会比较乱,可读性比较差,如果把其中一部分相同甚至可以复用的功能用一个新的函数来调用,然后让2个函数同时实现,就会做到

(1)进一步封装了函数的一些用法,做到dry原则(don't repeat yourself)

(2)使函数更加具有可读性

所以装饰器本身就是函数中调用其他函数,实现先拆分函数,再合并函数的功能。

2.函数的装饰器写法

(1)普通函数

这个函数实现的是计算2到9999的所有质数(在大于 1 的自然数中,除了 1 和它自身外,不能被其他自然数整除的数),并且打印找到这些数需要的时间。

会发现,这个time模块让整个代码逻辑很混乱,因为函数的主体是找质数,time模块是找质数的时间,如果可以time模块放在函数外,这样逻辑才清晰

import time# 是否为质数
def is_prime(n):if n <= 1 :return Falsefor i in range(2, int(n**0.5) + 1):if n % i == 0:return Falsereturn True# 定义一个函数,循环2到9999的数字,判断是否为质数
def prime_nums():t1 = time.time()for i in range(2, 10000):if is_prime(i):print(i)t2 = time.time()print(f"程序运行时间为:{t2 - t1}秒")# 调用函数
prime_nums()

(2)装饰器函数

装饰器的本质是一个高阶函数,它接收一个函数作为参数,并返回一个新函数来替代原函数。这个新函数需要:

a、保留原函数的调用方式(参数和返回值)。

b、在原函数执行前后添加额外逻辑(如计时、日志等)。

因此,我们需要在装饰器内部定义一个新函数来实现这些功能。

import time# 定义一个装饰器函数,用于计算函数的运行时间
def display_time(func):def wrapper():      # 定义一个内部函数,用于包装被装饰的函数,在装饰器中wrapper函数是一个常用的函数名start_time = time.time()  # 记录开始时间func()  # 直接调用原函数(无参数),这里的func()是指装饰器需要修饰的函数,在这里是prime_nums()end_time = time.time()  # 记录结束时间  print(f"函数 {func.__name__} 运行时间: {end_time - start_time} 秒")  # 打印函数运行时间# return wrapper是返回函数对象,如果是return wrapper()则是立即执行wrapper函数return wrapper  # 返回包装后的函数,这里的wrapper是指装饰器内部定义的函数,在这里是display_time()
# 继续定义判断质数的函数def is_prime(num):"""判断一个数是否为质数(处理负数和非整数输入)"""if num <= 1:  # 1和负数不是质数return Falsefor i in range(2, int(num**0.5) + 1):  # 只需要检查到平方根if num % i == 0:  # 如果能被整除,不是质数return Falsereturn True  # 否则是质数# 装饰器的标准写法
@display_time  # 应用装饰器,统计函数执行时间
def prime_nums():"""找到2到9999之间的所有质数并打印"""for i in range(2, 10000):  if is_prime(i):  # 如果是质数,打印print(i)# 调用函数,显示执行时间
prime_nums()
# 执行时间每次都会变,但是变动不大,一般计算稳定的执行时间我们都是重复1000遍,然后取平均

之所以采取这种写法可以实现这个逻辑,是因为装饰器在设计的时候底层思想如下,@display_time等价于

def prime_nums():...  # 函数体prime_nums = display_time(prime_nums) 

      装饰器的执行流程为:

a、定义装饰器函数 display_time:它接收一个函数 func 作为参数,并返回 wrapper 函数。

b、定义被装饰函数 prime_nums:此时 prime_nums 是一个普通函数对象。

c、应用装饰器:Python 自动将 prime_nums 作为参数传递给 display_time,即执行 display_time(prime_nums)

d、替换原函数display_time 返回 wrapper 函数,Python 用这个新函数覆盖了原来的 prime_nums

也就是说装饰后,原函数名指向 wrapper,而非原始函数。

       当你调用 prime_nums() 时,实际上执行的是 wrapper(),它会:

a、记录开始时间

b、调用 func()(即原函数)

c、记录结束时间并打印耗时

        这种等价的设计,会让初学者搞不懂为什么突然可以采取这种优雅的写法,类似的写法还有很多,在python中叫做语法糖:通过规范的写法来让代码更加优美和简洁。可以把@理解为语法糖操作,实际上并非是@装饰器,而是@装饰器+下一行的代码 二者是一个整体。

(3)语法糖

        语法糖(Syntactic Sugar)是编程语言中为简化代码书写、提升可读性而设计的语法特性。它本质是对底层逻辑的 “包装”,让开发者用更简洁的方式实现相同功能,但不会改变程序的实际行为或性能(即 “语法层面的优化”)。语法糖的核心目标是降低代码的 “心智负担”。通过更贴近自然语言或更紧凑的写法,让开发者用更少的代码表达相同逻辑,同时减少出错概率。

# 装饰器函数(统计耗时)
def timer(func):def wrapper(*args, **kwargs):start = time.time()res = func(*args, **kwargs)print(f"耗时:{time.time()-start:.2f}s")return resreturn wrapper# 无语法糖:手动绑定装饰器
def my_func():time.sleep(1)
my_func = timer(my_func)  # 必须显式赋值# 有语法糖:用 @ 符号直接标记(等价于上面的手动赋值)
@timer
def my_func():time.sleep(1)# 列表推导式 需求:生成一个包含 0-9 平方的列表
# 无语法糖:普通循环
squares = []
for x in range(10):squares.append(x**2)  # 3行代码# 有语法糖:列表推导式(1行代码)
squares = [x**2 for x in range(10)]# 三元表达式 需求:根据年龄判断是否成年
age = 20# 无语法糖:普通 if-else
if age >= 18:result = "成年"
else:result = "未成年"  # 4行代码# 有语法糖:三元表达式(1行代码)
result = "成年" if age >= 18 else "未成年"# with语句(上下文管理器) 需求:读取文件内容
# 无语法糖:手动 open/close(可能遗漏关闭)
f = open("test.txt", "r")
try:content = f.read()
finally:f.close()  # 必须显式关闭,否则可能占用资源# 有语法糖:with 语句(自动关闭文件)
with open("test.txt", "r") as f:content = f.read()  # 无需手动关闭,退出 with 块自动释放

3.注意内部函数的返回值

进一步拓展装饰器实现复用

可以看到,上述这个写法的时候,prime_nums()没有传入参数,如果函数有参数,那么必须给外部函数传入参数,也就是需要给外部的装饰器函数传入参数。

那么装饰器函数是需要复用的,不同的内部函数传入的参数不同,那就需要装饰器可以传入可变参数来维持这个特性。这就是说到了我们昨天的可变参数

装饰器函数返回的是wrapper函数,所以,在调用装饰器函数的时候,返回的还是wrapper函数,而不是被修饰的函数。他是被修饰函数的外层函数,参数要大于等于被修饰函数的参数

import time
def display_time(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  # 返回装饰器函数@display_time  # 应用装饰器到函数上,注意这里的括号不能少
def add(a, b):"""简单的加法函数"""return a + b  # 返回两个参数的和
add(3, 5) # 正常接收函数并计算

注意下内部函数有无返回值?

注意到之前被修饰的函数在无参数情况下,wrapper里面只有func(),现在是result = func(*args, **kwargs)以及加上了return result

为什么会这样?因为被修饰的函数是return xxxx,而不是print xxx,被修饰的函数如果有返回值,装饰器函数就需要搭配返回值。

4.作业:


编写一个装饰器 logger,在函数执行前后打印日志信息(如函数名、参数、返回值)
@logger
def multiply(a, b):
    return a * b

multiply(2, 3)  
# 输出:
# 开始执行函数 multiply,参数: (2, 3), {}
# 函数 multiply 执行完毕,返回值: 6

def logger(func):def wrapper(*args, **kwargs):   # args 是元组,kwargs 是字典print(f"调用函数: {func.__name__}")print(f"参数: {args}, {kwargs}")result = func(*args, **kwargs)print(f"函数 {func.__name__} 执行完毕,返回值: {result}")return resultreturn wrapper@logger
def multiply(a, b):return a * bmultiply(2, 3)  

@浙大疏锦行

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

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

相关文章

Qt进阶开发:动画框架的介绍和使用

文章目录 一、QPropertyAnimation 简介二、基本用法三、常用属性和方法四、支持的属性&#xff08;部分常用&#xff09;五、多个动画组合六、使用缓和曲线七、状态机框架 一、QPropertyAnimation 简介 #include <QPropertyAnimation>QPropertyAnimation 可以让你在一段…

IP选择注意事项

IP选择注意事项 MTP、FTP、EFUSE、EMEMORY选择时&#xff0c;需要考虑以下参数&#xff0c;然后确定后选择IP。 容量工作电压范围温度范围擦除、烧写速度/耗时读取所有bit的时间待机功耗擦写、烧写功耗面积所需要的mask layer

Flask RESTful 示例

目录 1. 环境准备2. 安装依赖3. 修改main.py4. 运行应用5. API使用示例获取所有任务获取单个任务创建新任务更新任务删除任务 中文乱码问题&#xff1a; 下面创建一个简单的Flask RESTful API示例。首先&#xff0c;我们需要创建环境&#xff0c;安装必要的依赖&#xff0c;然后…

filebeat原理架构

Filebeat 是基于 Golang 开发的轻量级日志采集 Agent&#xff0c;其核心架构设计围绕高效、可靠地采集与转发日志数据&#xff0c;主要组件和工作流程如下&#xff1a; ‌一、核心架构组件‌ ‌输入 (Inputs)‌ 负责监控指定的日志源&#xff08;如文件路径、日志文件&#x…

Air8000开发板新资料开放!多功能+高扩展特性全面解锁

Air8000开发板最新技术资料正式向开发者开放。这个开发板集多功能与高扩展性于一身&#xff0c;将为物联网、嵌入式系统等领域的创新项目提供更强大的技术支持&#xff0c;助力开发者快速实现创意落地。 工程师朋友们&#xff0c;Air8000开发板“多功能集成高扩展性”&#xff…

如何迁移Cordova应用到HarmonyOS 5 以及迁移时常见的问题?

以下是 Cordova 应用迁移至 HarmonyOS 5 的完整方案及常见问题解决方案&#xff0c;结合最新技术实践整理&#xff1a; 一、迁移流程 1. ‌方案选择‌ ‌方案‌‌适用场景‌‌操作复杂度‌‌Android 兼容层方案‌简单应用快速上线低&#xff08;无需修改代码&#xff09;‌原…

板凳-------Mysql cookbook学习 (十--4)

6.3 设置客户端时区 --客户端位于不同时区需要注意&#xff0c;如果位于同一时区则不需要关心 mysql> drop table if exists t; Query OK, 0 rows affected (0.06 sec)mysql> create table t (ts timestamp); Query OK, 0 rows affected (0.05 sec)mysql> insert int…

如何根据excel表生成sql的insert脚本

根据excel自带的vba宏进行操作 首先altF11 点击插入~模块 录取执行语句 Sub GenerateSQL()Dim lastRow As IntegerlastRow Cells(Rows.Count, 1).End(xlUp).RowFor i 2 To lastRow 假设第一行是标题Cells(i, "S").Value "INSERT INTO table_name (ID, RE…

React hook之useRef

React useRef 详解 useRef 是 React 提供的一个 Hook&#xff0c;用于在函数组件中创建可变的引用对象。它在 React 开发中有多种重要用途&#xff0c;下面我将全面详细地介绍它的特性和用法。 基本概念 1. 创建 ref const refContainer useRef(initialValue);initialValu…

开疆智能ModbusTCP转Canopen网关连接AB PLC与台达伺服通讯案例

本案例是罗克韦尔PLC通过开疆智能ModbusTCP转Canopen网关连接台达A2伺服的配置案例。 配置方法&#xff1a; 首先打开PLC配置软件“Studio5000”并新建项目导入通讯文件 对功能块进行设置 填写本地IP地址以及服务区IP地址以及寄存器 填写寄存器地址数量及使能 确认无误后将配置…

用 LoRA 对 Qwen2.5-VL 模型进行SFT - LoRA微调流程

用 LoRA 对 Qwen2.5-VL 模型进行SFT - LoRA微调流程 flyfish ┌──────────────────────────────────────────────────────────────────────────┐ │ 环境准备与启动 …

熵最小化Entropy Minimization (二): 案例实施

前面介绍了熵最小化、常用的权重函数汇总、半监督学习&#xff1a;低密度分离假设 (Low-Density Separation Assumption)、标签平滑、信息最大化等相关的知识点&#xff0c;本文采用一个MNIST10分类的数据集来进一步体会它们的效果。 案例实施 对比方法 纯监督学习方法&…

联邦学习聚合参数操作详解

联邦学习中常见的模型聚合操作&#xff0c;具体用于对来自多个客户端的模型更新进行聚合&#xff0c;以得到全局模型。在联邦学习框架下&#xff0c;多个客户端在本地训练各自的模型后&#xff0c;会将模型更新&#xff08;通常是模型的权重&#xff09;发送到中央服务器&#…

蓝牙 BLE 扫描面试题大全(2):进阶面试题与实战演练

前文覆盖了 BLE 扫描的基础概念与经典问题蓝牙 BLE 扫描面试题大全(1)&#xff1a;从基础到实战的深度解析-CSDN博客&#xff0c;但实际面试中&#xff0c;企业更关注候选人对复杂场景的应对能力&#xff08;如多设备并发扫描、低功耗与高发现率的平衡&#xff09;和前沿技术的…

基于Flask实现的医疗保险欺诈识别监测模型

基于Flask实现的医疗保险欺诈识别监测模型 项目截图 项目简介 社会医疗保险是国家通过立法形式强制实施&#xff0c;由雇主和个人按一定比例缴纳保险费&#xff0c;建立社会医疗保险基金&#xff0c;支付雇员医疗费用的一种医疗保险制度&#xff0c; 它是促进社会文明和进步的…

机房断电后 etcd 启动失败的排查与快速恢复实录

目录 机房断电后 etcd 启动失败的排查与快速恢复实录 背景与问题起因 报错分析 解决方案&#xff1a;删除 member 数据重新初始化 步骤 1&#xff1a;停止 etcd 容器或服务 步骤 2&#xff1a;删除 member 目录 步骤 3&#xff1a;重启 etcd 服务 附加提醒 总结与后续…

Linux上并行打包压缩工具

文章目录 Linux上并行打包压缩工具1. **Pigz (Parallel gzip)**2. **Pbzip2 (Parallel bzip2)**3. **Pixz (Parallel xz)**4. **Zstd (支持多线程)**5. **GNU Parallel 传统工具**6. **Fastest Tools (综合建议)**注意事项&#xff1a; tar和zstd多线程压缩和解压**1. 多线程压…

【K8S系列】K8S中Service 无法访问问题的深度分析

摘要 &#xff1a; 本文是关于 Kubernetes 中 Service 无法访问问题的深度分析&#xff0c;结合根本原因、诊断策略与解决方案的系统性指南&#xff1a; Kubernetes Service 访问故障全景解析&#xff1a;从底层机制到实战修复 Service 作为 Kubernetes 集群内网络通信的核心抽…

Leetcode 3577. Count the Number of Computer Unlocking Permutations

Leetcode 3577. Count the Number of Computer Unlocking Permutations 1. 解题思路2. 代码实现 题目链接&#xff1a;3577. Count the Number of Computer Unlocking Permutations 1. 解题思路 这一题其实就是一个脑筋急转弯&#xff0c;要想要能够将所有的电脑解锁&#x…

进行性核上性麻痹护理指南:全维度守护健康

进行性核上性麻痹是一种罕见的神经系统退行性疾病&#xff0c;会导致患者出现运动障碍、吞咽困难、认知障碍等症状。科学的健康护理对延缓病情发展、提升患者生活质量至关重要。 饮食护理&#xff1a;由于患者常存在吞咽困难&#xff0c;食物应选择糊状、软烂的类型&#xff0c…