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。在这个例子中,函数进入和退出时 ,被称为一个横切面,这种编程方式被称为面向切面的编程。

浙公网安备 33010602011771号