欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 房产 > 家装 > day 27 装饰器函数

day 27 装饰器函数

2025/6/8 12:54:31 来源:https://blog.csdn.net/weixin_47130140/article/details/148384628  浏览:    关键词:day 27 装饰器函数

一、装饰器基础概念

在 Python 项目中,当看到函数上方有 @xxx 这样的标识,它就是装饰器。装饰器本质上是一个 Python 函数,属于高阶函数,即接收一个函数作为参数,并返回一个新函数来替代原函数。其主要作用是在不修改目标函数代码的前提下,为函数或方法增加额外功能。

二、装饰器的优势

遵循 DRY 原则:当一个函数承担过多功能时,代码会显得混乱且可读性差。通过装饰器,可将部分相同或可复用的功能提取到新函数中,避免重复代码,实现 DRY (Don't Repeat Yourself)。

提高可读性:将特定功能从业务逻辑函数中分离,使业务逻辑函数更专注于核心功能,从而提升代码的整体可读性。

三、装饰器示例:计算质数并计时

普通函数实现

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()

此代码计时功能(使用 time 模块)与寻找质数的核心逻辑混在一起,使代码逻辑不够清晰。

装饰器实现

import time# 定义一个装饰器
def display_time(func):# 定义一个内部函数,在装饰器中wrapper函数是常用的函数名,并非强制,约定俗成。def wrapper():start_time = time.time()func()end_time = time.time()print(f"执000000行时间: {end_time - start_time} 秒")# 返回函数对象,如果是return wrapper()则是立即执行wrapper函数return wrapper# 继续定义判断质数的函数
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到10000之间的所有素数并打印"""for i in range(2, 10000):if is_prime(i):print(i)
prime_nums()

装饰器原理 

装饰器函数 display_time 接收函数 func 作为参数,内部定义了 wrapper 函数。wrapper 函数记录原函数执行前后的时间,然后打印执行时间。最后,display_time 返回 wrapper 函数对象。

执行流程

  1. 定义装饰器函数 display_time,它接收函数 func 作为参数,并返回 wrapper 函数。
  2. 定义被装饰函数 prime_nums,此时它是一个普通函数对象。
  3. 应用装饰器:当看到 @xxx 写在某个函数上方时,xxx 所代表的装饰器函数会将紧跟其后定义的函数作为 func 参数传入即执行 display_time(prime_nums)
  4. 替换原函数:display_time 返回 wrapper 函数,用这个新函数覆盖原来的 prime_nums。因此,调用 prime_nums() 时,实际执行的是 wrapper(),它会记录开始时间,调用原函数 func(),记录结束时间并打印耗时。

装饰器的语法糖与设计思想

在 Python 中,语法糖(Syntactic Sugar)指的是语言提供的一些特殊语法结构,这些结构使代码编写更加简洁、易读,但不会增加语言本身的功能。它们本质上是对常规操作的一种简洁表达方式,编译器或解释器会将其转换为底层的常规代码。@display_time 这种写法就是 Python 中的语法糖,它等价于:

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

在不使用语法糖时,我们要手动将函数传递给装饰器函数,并使用返回的新函数。而使用 @ 语法糖,代码看起来更简洁直观,Python 会自动帮我们完成函数传递和替换的操作。

带参数的装饰器与返回值处理

带参数的装饰器:如果被装饰函数需要传入参数,装饰器函数也需要相应地处理这些参数。为了使装饰器更具通用性,可使用可变参数 *args 和 **kwargs 接收任意数量的位置参数和关键字参数。

比如这里在 display_time 内部定义了 wrapper 函数。wrapper 函数使用 *args 和 **kwargs 作为参数,这样设计是为了让 wrapper 函数能够适配不同参数形式的被装饰函数。

import timedef display_time(func):"""支持任意参数的时间统计装饰器"""def wrapper(*args, **kwargs):t1 = time.time()result = func(*args, **kwargs)t2 = time.time()print(f"函数执行时间: {t2 - t1} 秒")return resultreturn wrapper@display_time
def add(a, b):return a + badd(3, 5)

返回值处理:当被装饰的函数有返回值时,装饰器内部的 wrapper 函数需要接收并返回原函数的返回值,以确保被装饰函数的返回值能正常传递给调用者,维持其原有的功能特性。

在上述示例中,add 函数返回两数之和,wrapper 函数通过 result = func(*args, **kwargs) 获取原函数返回值,并通过 return 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,参数: (2, 3), {}
函数 multiply 执行完毕,返回值: 6
6multiply(a=2, b=3)  
# 输出:
开始执行函数 multiply,参数: (), {'a': 2, 'b': 3}
函数 multiply 执行完毕,返回值: 6
6multiply(2, b=3)  
# 输出:
开始执行函数 multiply,参数: (2,), {'b': 3}
函数 multiply 执行完毕,返回值: 6
6multiply(a = 2, 3) 
# 报错,因为所有关键字参数必须跟在位置参数后面

@浙大疏锦行

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com

热搜词