1.1装饰器(decorator)
什么是装饰器:
器即函数
1.装饰器可以理解为给一个函数,做修饰,而不修改函数本身。
2.装饰器定义:本质就是函数,decorator功能就是为其他函数添加新的功能。
装饰器的的原则
装饰器=高阶函数+函数嵌套+闭包
装饰器的前提原则:不能修改被装饰函数的源代码,和函数的调用方式
1.1.1高阶函数
高阶函数定义:
1.函数接收的参数是一个函数名。
2.函数的返回值是一个函数。
3.满足上述条件任意一个,都是高阶函数
1.1.2函数嵌套
函数嵌套
python语言中的嵌套:定义一个函数的时候,函数体还能定义另一个函数。
在其他语言(例c语言),在一个函数调用另一个函数,叫嵌套
1.1.3闭包
存储在子封闭作用域(函数)的行为叫做闭包
一个闭包就是一个函数,只不过函数内部带上了一个额外的变量。
闭包关键特点就是它会记住自已被定义时的环境
1.2一步一步理解装饰器
1 defgo():2 print("python")3 #该函数功能执行go() ,显示出python
4 go()
现在我们要增强go()这个自定义函数的功能,比如在函数调用前自动打印出python是什么样的语言,但有不想在修改go()函数,这种在代码运行期间动态为其添加功能,就称之为装饰器(decorator)
1 defdeco(func):2 defwrapper():3func()4 print("一个高级语言")5 returnwrapper6@deco7 defgo():8 print("python")9go()10 #结果
11python12一个高级语言13
14 #现在才相当于一个合格的装饰器
15
16 把@deco放到go()函数的定义处,相当于执行了go=deco(go)
上面deco()是一个装饰器,返回一个函数,所以原来的go()函数仍然存在,
只是现在同名的go变量指向了新的函数,于是调用go()函数将执行新函数,
即在go()函数中返回的wrapper()函数
2.给被装饰器函数加参数和返回值
1 defdeco(func):2 def wrapper(*args,**kwargs):3 res = func(*args,**kwargs) #这里就相当于闭包
4 print("一门高级语言")5 returnres6 returnwrapper7
8@deco9 defgo(x,y):10 print("python",x,y)11 return "done"
12 go(3,5)13 #结果
14python15一个高级语言16
17 #为什么要给装饰器加参数,如果被装饰的函数里面有参数,我们的装饰器是不是器也要加相应的参数,
18 我们的装饰器为什么要给wrapper(*args,**kwargs) func(*args,**kwargs),可接收任意参数,因为我们被装饰的函数可能都是不同的的参数,而这个装饰器,需要给很多函数做装饰,但是很多函数的参数,功能都是不一样的,因此我们定义装饰器 的函数参数应该是加可变长参数19
20 #为什么给装饰里面加返回值
21 我们被装饰的函数,一般是有返回值,而执行装饰器(@decorator) 所以需要给wrapper 加上返回值来return fun()的执行结果,来保持被装饰的函数的一致性。
3.给装饰器加参数
如果装饰器本身需要传入参数,那就需要编写一个返回装饰器的高阶函数,也就是在原来装饰器上,做闭包处理,在加上一层函数。
def auth_book(auth=None):2 print(auth)3 defdeco(func):4 def wrapper(*args,**kwargs):5 res = func(*args,**kwargs) #这里就相当于闭包
6 print("一门高级语言")7 returnres8 returnwrapper9 returndeco10 @deco("book") #装饰器加参数 跟go=deco("book")(go)类似
11 defgo(x,y):12 print("python",x,y)13 go(3,5)
3层嵌套的装饰器的效果是这样的
go=auth_book("book")(go)
user_list=[2 {'name':'yj','passwd':'123'},3 {'name':'xixi','passwd':'123'},4 {'name':'xiha','passwd':'123'},5 {'name':'lala','passwd':'123'},6]7
8 current_user={'username':None,'login':False}9 def auth(auth_type='file'):10 defauth_deco(func):11 def wrapper(*args,**kwargs):12 if auth_type == 'file':13 if current_user['username'] and current_user['login']:14 res=func(*args,**kwargs)15 returnres16 username=input('用户名:').strip()17 passwd=input('密码:').strip()18
19 for index,user_dic inenumerate(user_list):20 if username == user_dic['name'] and passwd == user_dic['passwd']:21 current_user['username']=username22 current_user['login']=True23 res=func(*args,**kwargs)24 returnres25
26 else:27 print('用户名或者密码错误,重新登录')28 elif auth_type == 'ldap':29 print('巴拉巴拉小魔仙')30 res=func(*args,**kwargs)31 returnres32 returnwrapper33 returnauth_deco34
35
36 #auth(auth_type='file')就是在运行一个函数,然后返回auth_deco,所以@auth(auth_type='file')
37 #就相当于@auth_deco,只不过现在,我们的auth_deco作为一个闭包的应用,外层的包auth给它留了一个auth_type='file'参数
38 @auth(auth_type='ldap')39 defindex():40 print('欢迎来到主页面')41
42 @auth(auth_type='ldap')43 defhome():44 print('这里是你家')45 @auth(auth_type="file")46 defshopping_car():47 print('查看购物车啊亲')48
49 deforder():50 print('查看订单啊亲')51
52 #print(user_list)
53index()54 #print(user_list)
55home()56 shopping_car()