装饰器函数——用来对函数的扩展,补充。
高阶函数+嵌套函数 =》装饰器
原则:
1 / 不能修改被装饰的函数的源代码
2 / 不能修改被装饰的函数的调用方式
实现装饰器的知识储备:
1 / 函数即‘变量’
def test():
函数体
-----------------
test = '函数体'
2 / 高阶函数
两个条件:
1 / 把一个函数名当作实参传给另外一个函数(在不修改被装饰函数源代码的情况下为其添加功能)
2 / 返回值中包含函数名(不修改函数的调用方式)
import time def bar(): time.sleep(2) print('in the bar') def test2(func): print(func) return(func) # # print(test2(bar)) bar = test2(bar) bar()3 / 嵌套函数
装饰器的代码示例:
高阶函数+嵌套函数 =》装饰器
1 / 不能修改被装饰的函数的源代码
2 / 不能修改被装饰的函数的调用方式
由简到繁的代码说明装饰器:
——没有传递参数的代码示例讲解:
import time def timer(func):##装饰器 def deco(): start_time = time.time() func() end_time = time.time() print('the fun run time is %d' %(end_time-start_time)) return deco @timer ##语法糖 等价于test1 = timer(test1) def test1(): time.sleep(2) print('in the test1') test1() ''' 注解: 当test()调用时,实际因为语法糖的功能,等价于test1 = timer(test1),于是调用了装饰器timer, timer传入test1函数作为参数,运行装饰器下面的deco函数,执行deco函数,并return deco,注意这个deco没有小括号,返回的是deco的内存地址值。 最后调用test1()实际上是 “deco内存地址值()”在运行。 ''' ——被装饰函数需要传递参数的代码实力讲解: import time def timer(func):##装饰器 def deco(*args,**kwargs):##test2调用的时候传递了参数name,调用装饰器的时候,deco()也需要有参数name ##由于传递进来的参数不好确定,所以用*args,**kwargs接收 start_time = time.time() func(*args,**kwargs) end_time = time.time() print('the fun run time is %d' %(end_time-start_time)) return deco @timer ##语法糖 等价于test1 = timer(test1) def test1(): time.sleep(2) print('in the test1') @timer ##语法糖 等价于test2 = timer(test2) def test2(name): ##test2调用的时候传递了参数name,调用装饰器的时候,deco()也需要有参数name time.sleep(3) print('name:',name) test1() test2('drglon') ''' 注解: 当test2()调用时传入了参数name,实际因为语法糖的功能,等价于test2 = timer(test2(name)),于是调用了装饰器timer,并传入参数 timer传入test2(name)函数作为参数,运行装饰器下面的deco函数,并把参数name传递给deco函数, 执行deco(name)函数,并return deco,注意这个deco没有小括号,返回的是deco的内存地址值。 最后调用test2(name)实际上是 “deco内存地址值(name)”在运行。 因为装饰器中deco()函数需要接收的参数不固定,所以用*args,**kwargs接收。变的通用。 ''' ——关于被装饰函数的返回值如何体现 import time user,passwd = 'dralon','abc123' def auth(func): def wrapper(*args,**kwargs):##调用的时候有可能传递了参数,调用装饰器的时候,装饰器的函数也需要有参数 username = input('Username:').strip() password = input('Passwd:').strip() if user ==username and passwd == password: print('has authentication') res = func(*args,**kwargs) ##实际上home()的时候,home()的return home返回值需要在这里体现。设置一个变量储存home()函数的return home 返回值 print("这个是个home()的返回值 %s" %res) else: print('please input ') return wrapper def index(): print('welcom to index page') @auth def home(): print('welcom to home page') return home ##这个返回结果,在装饰器中要有返回才能体现出来 @auth def bbs(): print('welcom to bbs page') # index() home() # bbs()
高阶函数+嵌套函数 =》装饰器