Python高阶函数装饰器

简介: Python高阶函数装饰器

“ 从CANoe vTESTstudio版本7开始,支持使用python编辑器编写python脚本。其中CANoe提供了许多API接口给python使用,大大扩展了python的可用性。在python中使用装饰器定义capl中的事件处理程序(on key/on timer等)。对此我们有必要了解什么是装饰器”

装饰器,装饰是包装的意思,器表示工具。所以装饰器字面意思指的是包装用的工具。就像是买的礼物外面的包装盒子一样

可以看出,装饰器有三个特点:

不能改变礼物的本身
包装盒和礼物是一起的
拿出礼物时只会说礼物的名字,不会说包装盒的名称
“ 我们以给别人买生日蛋糕为例,你让服务员给蛋糕包装时,肯定不能让包装盒破坏蛋糕本身;你每次把蛋糕拿出来给别人看时,包装盒必定和蛋糕是一起拿出来的;当你拿出蛋糕时,只会介绍说这是蛋糕,并不会说这是我买的蛋糕和包装盒,对吧!”

那么在python中蛋糕和包装盒分别表示什么呢?蛋糕就是python函数,包装盒就是装饰器

所以,装饰器的特定是:

不能改变函数的内部代码
调用函数时装饰器一并调用
使用函数名调用函数
我们定义一个函数并运行:

def func1():
print("run func1")

func1()
打印的结果为:run func1

现在我想给它添加点特色,就是在打印“run func1”前先打印“program start”。有人说那我把func1函数体改成这样:

def func1():
print("program start")
print("run func1")

func1()
但是如此一来,就破坏了函数func1的函数体,这肯定不是装饰器的作用

那这样呢?

def func1():
print("run func1")

print("program start")
func1()
这样也不符合调用函数func1时一并调用装饰器,因为上面的代码实际上调用了print("program start")和func1(),肯定也不是装饰器的作用

那我利用函数的可参数化呢?

def func1():
print("run func1")

def prog1(func):
print("program start")
func()

prog1(func1)
这样使用的是其他函数调用的,也不符合装饰器的特点

分析:不能改变函数func1的结构,肯定得把函数func1当作参数传入另一个函数prog1中,在另一个函数中实现在调用func1前调用print("program start")。但是又不能通过调用prog1实现,还是要调用func1。怎么办?可以把prog1赋值给func1,以此实现调用func1即是调用prog1

下面这样可以吗?

def func1():
print("run func1")

def prog1(func):
print("program start")
func()

func1 = prog1(func1)

func1()
运行脚本后你会发现报错,报错内容如下:

出问题的点在哪呢?就在func1 = prog1(func1),当把调用prog1(func1)返回的对象赋值给func1时,调用prog1(func1)其实并没有返回值。那么func1就变成了None,最后调用func1肯定就报错了

所以需要在函数prog1的函数体内return一个对象,这个对象需要赋值给func1。由于func1是函数指针,那么prog1 return的对象也应该是一个函数指针。怎么办?可以在prog1函数体内再包一层函数,把这个函数指针返回

def func1():
print("run func1")

def prog1(func):
def wrapfunc():
print("program start")
func()
return wrapfunc

func1 = prog1(func1)

func1()
函数prog1这样的结构就是一个装饰器,使用时只需要把被装饰的函数指针传func1传参给prog1,然后再赋值给func1,如此调用func1就会连装饰器一起调用

但是func1 = prog1(func1)这样的写法麻烦且不具有共通性,所以python提供了一种装饰器的标准用法

def prog1(func):
def wrapfunc():
print("program start")
func()
return wrapfunc

@prog1
def func1():
print("run func1")

func1()
prog1是装饰器函数,func1是被装饰的函数,只需要在定义被装饰的函数前面用@符号引出装饰器函数名称,就实现了把被装饰的函数指针传给调用的装饰器函数然后赋值给被装饰函数指针的功能

结束了吗?并没有!

如果你觉得最后调用的func1还是以前的func1就错了,不信可以打印一下它的名称:

print(func1.name)
打印结果:wrapfunc

为什么?因为func1 = prog1(func1)时,调用prog1(func1)返回的就是wrapfunc函数,然后又赋值给了func1,所以func1已经不是之前的func1了。它重写了函数的名字和注释文档

有解决的办法吗?有!!!

python提供了一个函数functools.wraps解决这个问题

from functools import wraps

def prog1(func):
@wraps(func)
def wrapfunc():
print("program start")
func()
return wrapfunc

@prog1
def func1():
print("run func1")

func1()
@wraps接受一个函数来进行装饰,并加入了复制函数名称、注释文档、参数列表等等的功能。这可以让我们在装饰器里面访问在装饰之前的函数的属性

总结:

装饰器本质上是函数,需要编写
函数只有加上后面的括号才是调用,只有函数名,是函数指针
函数指针可以作为参数传参
函数指针作为对象可以赋值给其他的变量
函数体内可以定义其他的函数

目录
相关文章
|
16天前
|
数据安全/隐私保护 开发者 Python
深入浅出Python装饰器
【8月更文挑战第4天】装饰器在Python中是一个既强大又神秘的功能,它允许开发者在不修改原有函数代码的情况下增加额外的功能。本文旨在通过浅显易懂的语言和实例,带领读者一步步揭开装饰器的神秘面纱,理解其背后的原理,并学会如何在实际开发中应用它们。
Python 装饰器“高级”使用
本文聚焦两个有意思的点 1. 无参和有参装饰器。 @deco vs @deco(arg1,arg2)。 2. 多层装饰器场景。
|
16天前
|
Python
探索Python中的装饰器:从入门到实践
【8月更文挑战第4天】在Python的世界中,装饰器是一把双刃剑,它既能美化代码,又能提升效率。本文将带你一探究竟,通过实例学习如何定义、使用以及深入理解装饰器背后的原理。我们将一起揭开这层神秘的面纱,让装饰器成为你编程工具箱中的又一利器。
32 9
|
16天前
|
测试技术 开发者 Python
揭秘Python中的装饰器:从入门到精通
【8月更文挑战第4天】装饰器,在Python中是一块神奇的“画布”,它允许开发者在不修改原有函数代码的情况下增加额外的功能。本文将通过实际的代码示例,带你一探究竟,从基础使用到高级技巧,逐步揭开装饰器的神秘面纱。
|
14天前
|
测试技术 开发者 Python
翻天覆地!Python装饰器,如何让代码起死回生?
【8月更文挑战第6天】在软件开发领域,提高代码的质量始终是核心目标之一。Python作为一种功能丰富的高级语言,提供了多种手段来实现这一目标,装饰器便是其中之一。本文通过问答形式,深入解析了装饰器的概念、基本语法及其实现机制。装饰器允许在不改变原函数的基础上添加新功能,其基本语法为使用`@`符号后跟装饰器函数名。此外,还探讨了装饰器如何通过增强代码的模块性和灵活性来提升整体质量,并举例说明了装饰器在类方法中的应用。总之,装饰器是一种强大的工具,可以帮助开发者以更简洁、模块化的方式扩展功能,同时保持代码的整洁和可维护性。
25 3
|
17天前
|
设计模式 开发者 Python
探索Python中的装饰器:从基础到高级应用
【8月更文挑战第3天】本文将深入探讨Python编程中一个强大而灵活的特性——装饰器。我们将从理解装饰器的基本概念开始,通过实际代码示例,逐步展示如何创建和使用它们。文章旨在揭示装饰器背后的魔法,帮助开发者解锁其在代码重用和扩展性方面的潜力。
|
18天前
|
开发者 Python
Python编程中的装饰器深度解析
【8月更文挑战第2天】装饰器在Python中是一种强大的工具,它允许我们在不修改原函数代码的情况下增加函数的功能。本文将深入探讨Python装饰器的工作原理,并通过实际的代码示例展示如何创建和应用装饰器。我们将从基础的装饰器概念出发,逐步过渡到更复杂的使用场景,包括带参数的装饰器和嵌套装饰器。无论你是初学者还是有经验的开发者,这篇文章都将帮助你更好地理解和利用Python装饰器来提升你的代码效率和可读性。
11 1
|
8天前
|
数据处理 Python
Python使用装饰器记录函数执行时间
Python使用装饰器记录函数执行时间
18 0
|
13天前
|
分布式计算 Python
【python笔记】高阶函数map、filter、reduce
【python笔记】高阶函数map、filter、reduce