python-装饰器简述

装饰器是什么

用来修饰别的函数的函数就可以称之为装饰器

这种函数的参数一般就是另外一个函数

也就是说,调用这种函数,需要给这种函数传参,且参数是函数

@语法糖

@语法糖一般用来表示装饰器函数

不用@也可以达到装饰函数的目的,下面会有演示

函数嵌套

在一个函数中定义另外一个函数

def f1(arg="aaa"):
def f2():
return "hello"
def f3():
return "hi"
def f4():
return "haha" print f2()
print f3()
print f4()

这个f1函数有默认参数,所以可以不传参执行

执行f1()调用之后

执行结果如下:

hello

hi

haha

在函数里返回函数
def hi(arg="aaa"):
def greet():
return "bbb" def welcome():
return "ccc" if arg == "aaa":
return greet
else:
return welcome a = hi()
print(a)
print (a())

这里同样使用了默认参数,则a = hi() 会命中 if arg == "aaa"这个逻辑

返回greet,注意在这里,greet是函数,不是字符串,如果是返回字符串,则要返回的是 return "greet"这种

上面这段代码执行的结果是

<function greet at 0x7f75f7a0e1b8>

bbb

为什么是这样的执行结果呢?第一个地方 print(a),打印的是a = hi()的结果

我们可以看到,hi()的返回结果都是在return一个函数,要么是greet函数,要么是welcome函数

函数就是对应一个地址,所以第一处打印的是这个函数的地址

第二处做了a的调用,即a(),则打印返回的函数即greet函数执行的结果,即bbb

将函数作为参数传给另外一个函数
def hi():
return "hi" def hello(func):
print("before func()")
print(func()) hello(hi)

执行结果是

before func()

hi

这里把hi这个函数作为参数传给hello函数

hello函数先打印一句自身的输出before func()

再执行这个被传入的函数

我们可以看到,通过装饰器,我们可以在一个函数被调用前干一些需要的事情

不用@实现装饰器
def hi(a_func):

    def hello():
print("I am doing some boring work before executing a_func()") a_func() print("I am doing some boring work after executing a_func()")
return hello def haha():
print("I am the function which needs some decoration ") haha = hi(haha)
print haha()

运行结果:

I am doing some boring work before executing a_func()

I am the function which needs some decoration

I am doing some boring work after executing a_func()

None

这个就是装饰器,起到了用hi函数装饰了haha函数的功能

那么怎么用@语法糖实现呢
def hi(a_func):

    def hello():
print("I am doing some boring work before executing a_func()") a_func() print("I am doing some boring work after executing a_func()")
return hello @hi
def haha():
print("I am the function which needs some decoration ") haha()

可以看到,装饰器就是用希望装饰别的函数的函数,比如 ,希望用A装饰B

就在定义B函数的上一行,写上 @A

def A():
pass @A
def B():
pass

总结:理解就是装饰器其实就是这样一种函数:带参数的,且参数是另外一个函数的函数

使用装饰器的目的一般是为了在运行时改变函数的一些属性/行为,就可以给这个函数加上装饰器,让装饰器去在这个函数被调用前后,干一些你想做的事情

wraps干什么的?

我们修改下上面的代码,多记录一点东西

def hi(a_func):

    def hello():
print("I am doing some boring work before executing a_func()") a_func() print("I am doing some boring work after executing a_func()")
return hello @hi
def haha():
print("I am the function which needs some decoration ") haha()
print(haha.__name__)

我们的意图是打印haha这个函数的函数名,但实际上打印的是hello

为什么?

因为haha这个函数使用装饰器之后,haha的行为部分被装饰器函数改变了

可以看到,用hi函数装饰了haha之后,返回的是hello函数,那么当我们需要拿到被装饰函数的函数名,还有其他属性的时候怎么做呢?

使用functools.wraps方法

from functools import wraps
def hi(a_func):
@wraps(a_func)
def hello():
print("I am doing some boring work before executing a_func()") a_func() print("I am doing some boring work after executing a_func()")
return hello @hi
def haha():
print("I am the function which needs some decoration ") haha()
print(haha.__name__)

执行结果如下:

I am doing some boring work before executing a_func()

I am the function which needs some decoration

I am doing some boring work after executing a_func()

haha

正是我们想要的结果

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

举一个记录日志的例子
from functools import wraps

def logit(func):
@wraps(func)
def with_logging(*args, **kwargs):
print(func.__name__ + " was called")
return func(*args, **kwargs)
return with_logging @logit
def addition_func(x):
"""Do some math."""
return x + x result = addition_func(4)
# Output: addition_func was called
上一篇:多行文本超出后隐藏,超出3行隐藏


下一篇:RSA