python基础入门---函数装饰器

#装饰器:本质是函数,(装饰其它函数)就是为其他函数添加附加功能
#原则:1.不能修改被修饰的函数的源代码
# 2.不能修改被装饰的函数的调用方式
#实现装饰器知识储备:
# 1.函数即"变量"
# 2.高阶函数:
#a.把一个函数名当做实参传给另一个函数
#b.返回值中包含函数名
# 3.嵌套函数
#高阶函数+嵌套函数=》装饰器

更详细了解请点击:https://www.runoob.com/w3cnote/python-func-decorators.html

简单装饰器

def use_logging(func):

    def wrapper():
        logging.warn("%s is running" % func.__name__)
        return func()         # 把 foo 当做参数传递进来时,执行func()就相当于执行foo()
    return wrapper

def foo():
    print('i am foo')

foo = use_logging(foo)        # 因为装饰器 use_logging(foo) 返回的时函数对象 wrapper,这条语句相当于  foo = wrapper
foo()                        # 执行foo()就相当于执行 wrapper()


use_logging 就是一个装饰器,它一个普通的函数,它把执行真正业务逻辑的函数 func 包裹在其中,看起来像 foo 被
use_logging 装饰了一样,use_logging 返回的也是一个函数,这个函数的名字叫 wrapper。在这个例子中,函数进入
和退出时 ,被称为一个横切面,这种编程方式被称为面向切面的编程。

@ 语法糖

如果你接触 Python 有一段时间了的话,想必你对 @ 符号一定不陌生了,没错 @ 符号就是装饰器的语法糖,它放在函数开始定义的地方,
这样就可以省略最后一步再次赋值的操作。
def use_logging(func): def wrapper(): logging.warn("%s is running" % func.__name__) return func() return wrapper @use_logging def foo(): print("i am foo") foo() 如上所示,有了 @ ,我们就可以省去foo = use_logging(foo)这一句了,直接调用 foo() 即可得到想要的结果。
你们看到了没有,foo() 函数不需要做任何修改,
只需在定义的地方加上装饰器,调用的时候还是和以前一样,如果我们有其他的类似函数,我们可以继续调用装饰器来修饰函数,
而不用重复修改函数或者增加新的封装。这样,
我们就提高了程序的可重复利用性,并增加了程序的可读性。 装饰器在 Python 使用如此方便都要归因于 Python 的函数能像普通的对象一样能作为参数传递给其他函数,可以被赋值给其他变量,
可以作为返回值,可以被定义在另外一个函数内。


*args、**kwargs

可能有人问,如果我的业务逻辑函数 foo 需要参数怎么办?比如:

def foo(name):
    print("i am %s" % name)
我们可以在定义 wrapper 函数的时候指定参数:

def wrapper(name):
        logging.warn("%s is running" % func.__name__)
        return func(name)
    return wrapper
这样 foo 函数定义的参数就可以定义在 wrapper 函数中。这时,
又有人要问了,如果 foo 函数接收两个参数呢?三个参数呢?更有甚者,我可能传很多个。
当装饰器不知道 foo 到底有多少个参数时,我们可以用
*args 来代替: def wrapper(*args): logging.warn("%s is running" % func.__name__) return func(*args) return wrapper
如此一来,甭管 foo 定义了多少个参数,我都可以完整地传递到 func 中去。这样就不影响 foo 的业务逻辑了。
这时还有读者会问,如果 foo 函数还定义了一些关键字参数呢?比如:
def foo(name, age=None, height=None): print("I am %s, age %s, height %s" % (name, age, height))
这时,你就可以把 wrapper 函数指定关键字函数:
def wrapper(*args, **kwargs): # args是一个数组,kwargs一个字典 logging.warn("%s is running" % func.__name__) return func(*args, **kwargs) return wrapper

 



装饰器案列:

import time

def timer(func): #timer(test1)  func=test1
    def deco(*args,**kwargs):
        start_time = time.time()
        func(*args,**kwargs)    #run test1()
        stop_time = time.time()
        print("the func run time is %s" % (stop_time - start_time))
    return deco

@timer  #test1=timer(test1)= deco  test1()=deco()
def test1():
    time.sleep(1)
    print("in the t1")

@timer  ##test2=timer(test2)= deco  test2()=deco()
def test2(name,age):
    time.sleep(3)
    print("test2",name,age)

test1()
test2("qjh",23)

 

use_logging 就是一个装饰器,它一个普通的函数,它把执行真正业务逻辑的函数 func 包裹在其中,看起来像 foo 被 use_logging 装饰了一样,use_logging 返回的也是一个函数,这个函数的名字叫 wrapper。在这个例子中,函数进入和退出时 ,被称为一个横切面,这种编程方式被称为面向切面的编程。

posted @ 2020-03-27 12:21  七九灰  阅读(141)  评论(0)    收藏  举报