python decorator wrapper_python装饰器(decorator)

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()

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值