当前位置: 首页 > news >正文

Python训练营打卡 Day27

 函数专题2:装饰器

知识点回顾:

  1. 装饰器思想进一步复用
  2. 函数的装饰器写法
  3. 注意内部函数返回值

昨天我们接触到了函数大部分的功能,然后在你日常ctrl点进某个复杂的项目,发现函数上方有一个@xxx,它就是装饰器

装饰器本质上是一个 Python 函数,它可以让其他函数或方法在不需要做任何代码修改的前提下增加额外功能。--本质是如果让一个函数具备太多功能,那么他看起来就会比较乱,可读性比较差,如果把其中一部分相同甚至可以复用的功能用一个新的函数来调用,然后让2个函数同时实现,就会做到

装饰器可以看作是给函数“穿衣服”,让函数在执行时能有额外的功能,但又不改变函数本身的核心功能,同时要注意“衣服”怎么穿才合适,也就是内部函数返回值的问题。

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

2. 使函数更加具有可性

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

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

装饰器就像是餐厅的服务提升计划。餐厅(程序)有基本的出餐功能(函数),但为了提供更好的体验,可以增加一些额外的服务,比如上菜前的开胃小吃、上菜后的甜点等。这些额外服务可以复用在不同的菜品上,而不需要修改每道菜的制作流程。

普通的函数

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

1. 定义一个判断是否为质数

2. 定义一个函数,循环2到9999的数,通过判断质数函数来筛选每个数

3. 在函数中通过time模块进行记时

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

import timedef is_prime(num):if num < 2:return Falseelif num == 2:return Trueelse:for i in range(2, num):if num % i == 0:return Falsereturn Truedef 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()

函数的装饰器写法

假设我们有一个装饰器,用于记录每道菜的上菜时间,以提高服务效率。

import time# 装饰器函数:记录上菜时间
def timing_decorator(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:.2f} 秒")return result  # 返回上菜的结果return wrapper# 使用装饰器
@timing_decorator
def order_dish(dish):print(f"开始制作 {dish}...")time.sleep(2)  # 模拟制作时间print(f"{dish} 上菜啦!")return f"{dish} 完成"# 调用被装饰的函数
result = order_dish("宫保鸡丁")
print(result)

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

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

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

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

# 继续定义判断质数的函数
def is_prime(num):"""判断一个数是否为素数"""if num < 2:return Falseelif num == 2:return Trueelse:for i in range(2, num):if num % i == 0:return Falsereturn True# 装饰器的标准写法
@display_time
def prime_nums(): # 这2行是一个整体"""找出2到10000之间的所有素数并打印"""for i in range(2, 10000):if is_prime(i):print(i)prime_nums()
# 执行时间每次都会变,但是变动不大,一般计算稳定的执行时间我们都是重复1000遍,然后取平均

内部函数的返回值

在上面的例子中,wrapper 函数是装饰器的内部函数,它负责执行原函数 func 并记录时间。注意,wrapper 函数必须返回 func 的结果(return result),否则调用被装饰的函数时将无法获取到原函数的返回值。

就像服务员在上菜后,必须把菜端给顾客,否则顾客就吃不到菜。如果 wrapper 不返回 result,顾客(调用者)就无法获得上菜的结果。

总结

装饰器就像是给函数“穿衣服”,通过装饰器可以在不修改原函数的情况下,增加额外的功能。而内部函数的返回值就像是“服务员端菜”,必须确保把结果返回给调用者,否则就失去了原函数的意义。

作业

编写一个装饰器 logger,在函数执行前后打印日志信息(如函数名、参数、返回值)

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

@浙大疏锦行

相关文章:

  • 青少年编程与数学 02-019 Rust 编程基础 15课题、错误处理
  • 基于Mongodb的分布式文件存储实现
  • 旧 docker 版本通过 nvkind 搭建虚拟多节点 gpu 集群的坑
  • Ubuntu 22.04 上安装 Drupal 10并配置 Nginx, mysql 和 php
  • Spark,SparkSQL操作Mysql, 创建数据库和表
  • uni-app 中适配 App 平台
  • (面试)Android各版本新特性
  • 还没用过智能文档编辑器吗?带有AI插件的ONLYOFFICE介绍
  • 基于OpenCV的SIFT特征匹配指纹识别
  • 【第76例】IPD流程实战:华为业务流程架构BPA进化的4个阶段
  • 架构与UML4+1视图
  • 如何将数据从一部手机传输到另一部手机 | 5 种便捷传输方式
  • linux 服务器安装jira-8.22.0和confluence-8.5.21
  • 排序算法之高效排序:快速排序,归并排序,堆排序详解
  • nginx Permission denied
  • virtualbox虚拟机中的ubuntu 20.04.6安装新的linux内核5.4.293 | 并增加一个系统调用 | 证书问题如何解决
  • 反向传播算法:神经网络的核心优化方法,一文打通任督二脉
  • Excel MCP: 自动读取、提炼、分析Excel数据并生成可视化图表和分析报告
  • 车道线检测----CLRKDNet
  • 5.9/Q1,GBD数据库最新文章解读
  • 原核试验基地司令员范如玉逝世,从事核试验研究超40年
  • 61岁云浮市律师协会副会长谭炳光因突发疾病逝世
  • 外企聊营商|上海仲裁:化解跨国企业纠纷的“上海路径”
  • 词条数量大幅扩充,《辞海》第八版启动编纂
  • 美国务卿会见叙利亚外长,沙特等国表示将支持叙利亚重建
  • 大陆非遗项目打铁花、英歌舞将在台演出