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   
posted @ 2018-04-06 11:17  aaa1111sss  阅读(448)  评论(2编辑  收藏  举报