Python的装饰器,是一个Python中一个比较难以理解的知识点, 今天我试试说一说,说的如果不好,请见谅。
装饰器,从字面意思来说,就是修饰一个事物的,在Python里面的作用就是让一个已经存在的函数拥有一个以前没有的功能。装饰器本质上还是一个函数,知识他的功能是为其他函数添加新功能。
那有人会说,我在原来的函数里面加一些代码,也可以实现啊,为什么还要搞个装饰器来增加功能呢? 当然,如果你给一个函数加个功能,是可以加代码, 但是如果是让你给100个函数代码块加一个相同的功能呢? 一个个改代码吗? 你会奔溃的。
下面说说一个装饰器要实现,要符合一定的规则,就是两个不改动
1.不改动被装饰函数的源代码
2.不改动被修饰函数的调用方式
就是你加没加这个装饰器,函数要按照原样来调用
下面说说装饰器到底怎么写,怎么用,上面说到装饰器也是一个函数,但是这个函数和普通的函数不太一样,他是一个高阶函数,那我们给装饰器大概下个定义
装饰器:一个可以给别的函数添加新功能的高阶函数(当然现在还是不完整的定义)
下面介绍下什么是高阶函数。
高阶函数: 一个可以接受函数为参数或者能reture 一个函数的函数就是高阶函数
是不是有点绕,举个例子
def f1():
print('普通函数')
def f2(func): #f2函数的参数是一个函数,所以它是一个高阶函数
print('高阶函数1')
f1()
def f3(func): #f3函数的返回值是一个函数,所以它是一个高阶函数
print('高阶函数2')
return func
f2(f1)
f3(f1)
结果:
高阶函数1
普通函数
高阶函数2
有点基础的人可能知道,这个根本就不是装饰器,骗人的。当然装饰器不是这么简单。
那下面要介绍的一个东西叫嵌套函数,那什么叫嵌套函数呢? 字面意思应该就知道了,函数里面套一个子函数
举个例子:
def outer():
print('outer')
def inner():
print('inner')
inner()
outer()
结果
outer
inner
下面我们要给装饰器重新下个定义了。
装饰器:一个可以给别的函数添加新功能的高阶函数+嵌套函数
那么我们把高阶函数+嵌套函数结合在一起看看
一个可以添加一行日志的装饰器
def add_log(func): #一个函数作为参数,这个函数就是我们要
def wrapper():
print('我们要加一些日志') #要给原函数添加的新功能,这里就是打印一行日志,当然可以实现其他复杂的功能
func()
return wrapper
#上面可以看到高阶函数+嵌套函数的影子吧,加一起就是一个简单的装饰器拉
@add_log #装饰器的用法
def f1():
print('普通函数')
f1() #调用函数
结果:
#看结果f1里面只有print一行,但是结果多了一行,就是装饰器的作用
我们要加一些日志
普通函数
这些装饰器大概知道上面意思了吧。
下面说说,上面的装饰器,有没有没有上面问题呢?
看看,仔细看看。 看出来了吗? 看不出来,我说了哦。
上面我们的f1()函数, 没有参数,是不是,想想如果f1()函数要加参数怎么办啊?
def add_log(func):
def wrapper(name):
print('我们要加一些日志')
func(name)
return wrapper
@add_log
def f1(name):
print('普通函数 %s' %name)
f1('python')
有人是不是想这样实现呢?
当然这样是可以的,但是我们不要忘了装饰器的作用,他不是给某一个固定的函数来使用的,假如现在还需要修饰一个f2(name,age), 那怎么办啊? 没办法,装饰器的代码都给你定死了,所以上面这样写是不行的。
到底怎么写呢?
def add_log(func):
def wrapper(*args,**kwargs):
print('我们要加一些日志')
func(*args,**kwargs)
return wrapper
@add_log
def f1(name):
print('普通函数 %s' %name)
@add_log
def f2(name,age):
print('我是%s,我%s岁了' %(name,age))
f1('python')
f2('java',10)
结果:
我们要加一些日志
普通函数 python
我们要加一些日志
我是java,我10岁了
上面的写法,是不是可以了,是不是,不管任何函数,任何参数,都OK了,至于args,*kwargs啥意思,大家可以去百度下啊。这里就不多说了
今天就说这么多,希望大家对于python的装饰器有个基本的了解。
链接:https://juejin.im/post/5df39ed55188251227530b3f
来源:掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。