Python装饰器1-闭包与函数装饰器

简介: 闭包与函数装饰器:被装饰函数不带参数、被装饰函数带参数、装饰器带参数,装饰器的调用

一、闭包

在学习装饰器前,需要先了解闭包的概念。

形成闭包的要点:

  • 函数嵌套
  • 将内部函数作为外部函数的返回值
  • 内部函数必须要使用到外部函数的变量

下面以一个计算列表平均值的案例来讲解闭包:

defmake_average():
# 创建一个列表,用来保存数值nums= []
# 定义一个内部函数,用来计算列表的平均值defaverage(n):
# 将数值添加到列表中nums.append(n)
# 返回平均值returnsum(nums) /len(nums)
returnaverage
  1. 首先,定义一个函数make_average;
  2. 其次,在make_average函数内定义一个空列表,用来存储数值;
  3. 再次,定义一个内部函数,用来计算列表平均值;
  4. 最后,将这个内函数作为外函数make_average的返回值,注意不要加( ),加( )就变成调用这个函数了。
# 调用外部函数,并将其复制给一个变量,注意:此时返回的是内函数的内存地址a=make_average()
# 给这个变量加(),就相当于调用了内函数averageprint(a(20))
print(a(30))

运行结果如下:当传入的数值为20时,列表中只有一个数,所以计算结果是20;当再传入一个数值30时,此时列表中就有两个数20和30,所以平均值的计算结果是25.

二、装饰器

1.装饰器引入

例如,有以下两个函数,分别计算两个数的和以及成绩:

defadd(a, b):
"""计算两数之和"""res=a+breturnresdefmul(a, b):
"""计算两数之积"""res=a*breturnres

现在有个需求:我想要在每个函数的计算开始前打印“开始计算...”,在计算结束后打印“计算结束...”。我们可以通过直接修改函数代码的方式来满足这个需求,但这样会面临以下问题:

  1. 如果要修改的函数过多,十个甚至一百个函数,未免不现实;
  2. 不便于后期维护,例如我不想打印“开始计算...”了,而是要打印“begin...”,岂不是又要重新修改一遍;
  3. 违反开闭原则(OCP),即程序的设计,要求开放对程序的扩展、关闭对程序的修改;

所以,上述直接修改函数代码的方式不可行。我们希望在不修改原函数的情况下,实现对函数的扩展。例如:

defnew_add(a, b):
print("开始计算...")
r=add(a, b)
print("计算结束...")
returnrprint(new_add(22, 33))

执行结果如下:

这种新创建一个函数的方式虽然没有修改原函数,但面临一个很严重的问题:

只能扩展指定函数,不能通用于其它函数,例如扩展上述的add函数,而不能扩展mul函数,如果想要扩展mul函数,只能再创建一个扩展函数;

因为,我们希望可以定义一个通用的扩展函数,可以作用域所有函数。这类不改变原函数代码的通用函数就是:装饰器

2.函数装饰器

装饰器本质上是一个python函数或类,它可以让其他函数或类在不需要做任何代码修改的前提下增加额外功能,也就是为已经存在的对象添加额外功能,装饰器的返回值也是一个函数/类对象。它经常用于有切面需求的场景,比如:插入日志、性能测试、事务处理、缓存、权限校验等场景。

1)被装饰函数不带参数

例如:

defwrapper_info(func):
definner():
print("开始介绍...")
res=func()
print("介绍结束...")
returnresreturninnerdefintroduce1():
print("我是周润发,我来自HONG KONG")
info=wrapper_info(introduce1)
info()

运行结果如下:

可见,在没有改变原函数代码的情况下,即给原函数增加了一些额外的功能,func是要修饰的函数,作为一个变量传入装饰函数,能够通用于其他函数,这个wrapper_info就是装饰器。但目前面临的问题是,被装饰函数如果带参数怎么办?例如:

defintroduce2(name, age):
print(f"我叫{name}, 我今年{age}岁了")

2)被装饰函数带参数

尽管可以在装饰器wrapper_info中传入name、age,但并不是每个被装饰的函数都只有name、age,亦或是指定类型的参数,还有可能传入的是字典、列表、元组等。也就是传入参数的类型和数量不固定怎么办?

此时就需要用到不定长参数:(*args, **kwargs)

defwrapper_info(func):
"""    用来对其他函数进行扩展,使其他函数可以在执行前做一些额外的动作    :param func: 要扩展的函数对象    :return:    """definner(*args, **kwargs):
print("开始介绍...")
res=func(*args, **kwargs)
print("介绍结束...")
returnresreturninner

例如:

defintroduce3(name, age, city):
print(f"我叫{name}, 我今年{age}岁了, 我来自{city}")

运行结果如下:

3)装饰器带参数

上述提到的是装饰器,一种是应用于被装饰的函数不带参数,一种是被装饰的函数带参数,那装饰器本身能否带参数呢?比如我定义一个变量,想通过传入不同的值来控制这个装饰器实现不同的功能。答案是肯定的,例如:

defuse_log(level):
defdecorator(func):
definner(*args, **kwargs):
iflevel=="warn":
logging.warning("%s is running by warning"%func.__name__)
eliflevel=="info":
logging.warning("%s is running by info"%func.__name__)
else:
logging.warning("%s is running by other"%func.__name__)
returnfunc(*args, **kwargs)
returninnerreturndecoratordefintroduce4(name, age, city):
print(f"我叫{name}, 我今年{age}岁了, 我来自{city}")
info1=use_log(introduce4('周星驰', 28, '香港'))
info1('info')
info2=use_log(introduce4('周润发', 28, '香港'))
info2('warn')
info3=use_log(introduce4('成龙', 28, '香港'))
info3('xxx')

运行结果如下:

3.装饰器调用

方式一:以函数方式调用

info3=wrapper_info(introduce3)
info3('刘德华', 28, '香港')

如果是装饰器函数带参数,则调用方式为:

info4=use_log(introduce4('周星驰', 28, '香港'))
info4('info')

方式二:以语法糖方式调用

即在被装饰函数上方以@符号进行修饰

@wrapper_infodefintroduce3(name, age, city):
print(f"我叫{name}, 我今年{age}岁了, 我来自{city}")
introduce3('刘德华', 28, '香港')

如果是装饰器函数带参数,例如上述的use_log,则需要在装饰器中传入参数:

@use_log('info')
defintroduce4(name, age, city):
print(f"我叫{name}, 我今年{age}岁了, 我来自{city}")

小结

什么是装饰器?

在不改变原函数代码的情况下,给原函数增加了一些额外的功能,并且能够通用于其他函数,这样的函数就称作为装饰器

装饰器的调用

可以通过传统调用函数的方式进行调用,也可以通过@装饰器的方式调用

装饰器的特点

  • 通过装饰器,可以在不修改原来函数的情况下对函数进行扩展
  • 一个函数可以同时指定多个装饰器




目录
打赏
0
0
0
0
843
分享
相关文章
Python入门:8.Python中的函数
### 引言 在编写程序时,函数是一种强大的工具。它们可以将代码逻辑模块化,减少重复代码的编写,并提高程序的可读性和可维护性。无论是初学者还是资深开发者,深入理解函数的使用和设计都是编写高质量代码的基础。本文将从基础概念开始,逐步讲解 Python 中的函数及其高级特性。
Python入门:8.Python中的函数
[oeasy]python061_如何接收输入_input函数_字符串_str_容器_ 输入输出
本文介绍了Python中如何使用`input()`函数接收用户输入。`input()`函数可以从标准输入流获取字符串,并将其赋值给变量。通过键盘输入的值可以实时赋予变量,实现动态输入。为了更好地理解其用法,文中通过实例演示了如何接收用户输入并存储在变量中,还介绍了`input()`函数的参数`prompt`,用于提供输入提示信息。最后总结了`input()`函数的核心功能及其应用场景。更多内容可参考蓝桥、GitHub和Gitee上的相关教程。
16 0
Python装饰器是什么?
装饰器是Python中用于动态修改函数、方法或类功能的工具,无需改变原代码。通过将函数作为参数传递并返回新函数,装饰器可以在原函数执行前后添加额外逻辑。例如,使用`@logger`装饰器可以打印函数调用日志,而`@timethis`则可用于计算函数执行时间。为了保持被装饰函数的元信息(如`__name__`和`__doc__`),可使用`functools.wraps`装饰器。此外,带参数的装饰器可通过嵌套函数实现,如`@timeitS(2)`,以根据参数条件输出特定信息。
90 59
Python装饰器实战:打造高效性能计时工具
在数据分析中,处理大规模数据时,分析代码性能至关重要。本文介绍如何使用Python装饰器实现性能计时工具,在不改变现有代码的基础上,方便快速地测试函数执行时间。该方法具有侵入性小、复用性强、灵活度高等优点,有助于快速发现性能瓶颈并优化代码。通过设置循环次数参数,可以更准确地评估函数的平均执行时间,提升开发效率。
106 61
Python装饰器实战:打造高效性能计时工具
|
1月前
|
[oeasy]python057_如何删除print函数_dunder_builtins_系统内建模块
本文介绍了如何删除Python中的`print`函数,并探讨了系统内建模块`__builtins__`的作用。主要内容包括: 1. **回忆上次内容**:上次提到使用下划线避免命名冲突。 2. **双下划线变量**:解释了双下划线(如`__name__`、`__doc__`、`__builtins__`)是系统定义的标识符,具有特殊含义。
32 3
|
1月前
|
深入理解 Python 的 eval() 函数与空全局字典 {}
`eval()` 函数在 Python 中能将字符串解析为代码并执行,但伴随安全风险,尤其在处理不受信任的输入时。传递空全局字典 {} 可限制其访问内置对象,但仍存隐患。建议通过限制函数和变量、使用沙箱环境、避免复杂表达式、验证输入等提高安全性。更推荐使用 `ast.literal_eval()`、自定义解析器或 JSON 解析等替代方案,以确保代码安全性和可靠性。
45 2
python装饰器底层原理
Python装饰器是一个强大的工具,可以在不修改原始函数代码的情况下,动态地增加功能。理解装饰器的底层原理,包括函数是对象、闭包和高阶函数,可以帮助我们更好地使用和编写装饰器。无论是用于日志记录、权限验证还是缓存,装饰器都可以显著提高代码的可维护性和复用性。
50 5
Seaborn 教程-绘图函数
Seaborn 教程-绘图函数
87 8
Python中的装饰器深度解析与实践
在Python的世界里,装饰器如同一位神秘的魔法师,它拥有改变函数行为的能力。本文将揭开装饰器的神秘面纱,通过直观的代码示例,引导你理解其工作原理,并掌握如何在实际项目中灵活运用这一强大的工具。从基础到进阶,我们将一起探索装饰器的魅力所在。
探索Python中的装饰器:简化代码,增强功能
在Python的世界里,装饰器就像是给函数穿上了一件神奇的外套,让它们拥有了超能力。本文将通过浅显易懂的语言和生动的比喻,带你了解装饰器的基本概念、使用方法以及它们如何让你的代码变得更加简洁高效。让我们一起揭开装饰器的神秘面纱,看看它是如何在不改变函数核心逻辑的情况下,为函数增添新功能的吧!

热门文章

最新文章

推荐镜像

更多
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等