越来越觉得写一点技术博客是有多么重要了,明日复明日,现在就开始写吧!

1. 普通装饰器

  装饰器的写法是一种语法糖,装饰器也还是一个函数而已,它接收一个函数对象作为参数,并返回一个新函数,主要是拓展原函数功能,在不改变原函数代码结构的情况下附加操作达到业务需求。比如做身份认证、访问记录等预处理操作,或者是日志记录、通道关闭等收尾操作,又或者是性能测试这种前后都附加代码的操作。

  直接就上一个的基本的装饰器代码  (用print语句代替功能实现)

def deco(fun):
def inner(*args,**kwargs):
print('Successful authentication') # 预处理操作,验证身份
fun(*args,**kwargs) # 原操作
return inner @deco
def func(*args,**kwargs):
print(args,kwargs)  func(1,2,a=3)
# Perform pretreatment
# (1, 2) {'a': 3}

  当调用func时,伪流程就是:

    1. 先记录了func接收的参数们的内存地址

    2. 将func函数对象作为参数传入来调用deco,执行时将参数func绑定到inner函数对象上,返回inner函数对象

    3. 调用inner函数,传入之前记录的参数们,执行预操作(验证身份)

    4. 调用func函数,传入inner接收的参数们,执行原操作

  相当于合成并调用了一个新函数,新函数里调用了旧函数。但是好像我们去掉装饰器这个标记语法糖,还原为两个正常的函数,也能实现上面的逻辑吧:

def deco(fun):
def inner(*args,**kwargs):
print('Successful authentication') # 预处理操作,验证身份
fun(*args,**kwargs)  # 原操作
return inner def func(*args,**kwargs):
print(args,kwargs)  

new_fun = deco(func)
  new_fun(1,2,a=3)
  # Perform pretreatment
  # (1, 2) {'a': 3}

  但真实的流程是装饰器在文件载入的时候,就已经处理了被装饰器标记的函数,也就是说当文件载入时,所有被标记的函数都已经变成了装饰器返回的函数了(各自信息唯一),当调用的时候也就直接调用的此函数了。

2.装饰链

  也就是多层装饰器嵌套了一下

def deco1(fun):
def inner1(*args,**kwargs):
print('Successful authentication') # 预处理操作,验证身份
fun(*args,**kwargs)
return inner1 def deco2(fun):
def inner2(*args,**kwargs):
print('Successful access record') # 预处理操作,访问记录
fun(*args,**kwargs)  # 原操作
return inner2 @deco1
@deco2
def func(*args,**kwargs):
print(args,kwargs) func(1,2,a=3)
# Perform pretreatment
# (1, 2) {'a': 3}

  需要注意的是包装顺序,距离最远的是最外层的包装,就像穿的衣服一样,要先穿贴身的内衣,然后才是外套什么的

  此时调用func时,基本的流程就是:

    1. 记录了func接收的参数们的内存地址

    2. 将func函数对象作为参数传入来调用deco2,执行时将参数func绑定到inner2函数对象上,返回inner2函数对象

    3. 将inner2函数对象作为参数传入来调用deco1,执行时将参数inner2绑定到inner1函数对象上,返回inner1函数对象

    4. 调用inner1函数,传入之前记录的参数们,执行预操作1(验证身份)

    5. 调用inner2函数,传入inner1接收的参数们,执行预操作2(访问记录)

    6. 调用func函数,传入inner2接收的参数们,执行原操作

3.类装饰器

类的__call__方法可以让类实例可以像函数一样调用,所以跟普通的区别不大

class Deco(object):
def __init__(self, fun):
self.fun = fun
def __call__(self,*args,**kwargs):
print("Successful authentication") # 预处理 身份验证
self.fun(*args,**kwargs) @Deco
def func(*args,**kwargs):
print("func") func(1,2,a=3)
# Perform pretreatment
# (1, 2) {'a': 3}

4. 装饰器参数

  利用闭包绑定了参数

def deco_param(*dargs,**dkwargs)
def deco(fun):
def inner(*args,**kwargs):
print('deco_param:',dargs,dkwargs) # 打印装饰器参数
print('Successful authentication') # 预处理操作,验证身份
fun(*args,**kwargs)
return inner
return deco @deco(4,b=5)
def func(*args,**kwargs):
print(args,kwargs)  # 原操作 func(1,2,a=3)
# Perform pretreatment
# (1, 2) {'a': 3}

  

5.wraps恢复原函数信息

 按照之前的第一个装饰器例子,打印原函数名,发现被装饰的函数信息已经被丢失,变成了装饰器返回的函数的信息

def deco(fun):
def inner(*args,**kwargs):
print('Successful authentication') # 预处理操作,验证身份
fun(*args,**kwargs)
return inner @deco
def func(*args,**kwargs):
print(args,kwargs)  # 原操作
print(func.__name__)  
# inner

  要想恢复函数之前的信息,那么就需要用上functools包,在闭包函数上加上一个带参的装饰器,逻辑上只是加长了装饰链,而这个装饰器的拓展功能正是找回原函数(多层装饰器中可以是上一个装饰器的返回函数)的信息

from functools import wraps
def deco(fun):
  @wraps(fun)
def inner(*args,**kwargs):
print('Successful authentication') # 预处理操作,验证身份
fun(*args,**kwargs)
return inner @deco
def func(*args,**kwargs):
print(args,kwargs)  # 原操作 print(fun.__name__)
# func

  ps:一般情况不用找回函数信息,但在使用flask时,flask中不允许接口函数重名导致覆盖或被覆盖,如果有多个接口使用同一个装饰器的话,正常情况下接口函数信息就都为装饰器中的闭包函数信息,这样flask会报AssertionError: View function mapping is overwriting an existing endpoint function xx.inner的错误,此时就可以用上functools的wraps装饰器来找回原函数信息了。而flask必带的接口装饰器route查看源码似乎没有用到wraps,但是肯定也是做了处理的。

以上就是一些装饰器的介绍,如有错误,还请指出-----1738268742@qq.com

python_装饰器的更多相关文章

  1. python_装饰器_语法糖

    什么是高阶函数? -- 把函数名当做参数传给另外一个函数,在另外一个函数中通过参数调用执行 #!/usr/bin/python3 __author__ = 'beimenchuixue' __blog ...

  2. Python_装饰器进阶_32

    #带参数的装饰器 #500个函数 import time FLAGE = True def timmer_out(flag): def timmer(func): def inner(*args,** ...

  3. Python_装饰器习题_31

    # 1.编写装饰器,为多个函数加上认证的功能(用户的账号密码来源于文件), # 要求登录成功一次,后续的函数都无需再输入用户名和密码 FLAG = False def login(func): def ...

  4. Python_装饰器复习_30

    复习: # 装饰器的进阶 # functools.wraps # 带参数的装饰器 # 多个装饰器装饰同一个函数# 周末的作业 # 文件操作 # 字符串处理 # 输入输出 # 流程控制 # 装饰器# 开 ...

  5. Python_装饰器_29

    # 装饰器形成的过程 : 最简单的装饰器 有返回值的 有一个参数 万能参数 # 装饰器的作用 # 原则 :开放封闭原则 # 语法糖 :@ # 装饰器的固定模式 import time # print( ...

  6. python_装饰器模式

    装饰器模式定义:动态地给一个对象添加一些额外的职责. 在Python中Decorator mode可以按照像其它编程语言如C++, Java等的样子来实现,但是Python在应用装饰概念方面的能力上远 ...

  7. Python_装饰器、迭代器、生成器

    一.装饰器 装饰器的存在是为了实现开放封闭原则: 封闭: 已实现的功能代码块不应该被修改: 开放: 对现有功能的扩展开放. 理解装饰器的三要素: 函数的作用域 高阶函数 闭包 1. 闭包 闭包定义:如 ...

  8. python_装饰器——迭代器——生成器

    一.装饰器 1.什么是装饰器? 器=>工具,装饰=>增加功能 1.不修改源代码 2.不修改调用方式 装饰器是在遵循1和2原则的基础上为被装饰对象增加功能的工具 2.实现无参装饰器 1.无参 ...

  9. Python_装饰器函数

    楔子 作为一个会写函数的python开发,我们从今天开始要去公司上班了.写了一个函数,就交给其他开发用了. def func1(): print('in func1') 季度末,公司的领导要给大家发绩 ...

随机推荐

  1. HtmlUnit学习总结

    HtmlUnit学习总结 转载 2016年09月13日 15:58:25 标签: htmlunit / 爬虫 7304 本文摘抄其他博客或者技术论坛,自己搜集整理如下: HtmlUnit学习总结 摘要 ...

  2. spring定时器cron

    关于cron表达式(参考资料):Cron 表达式包括以下 7 个字段: 秒 分 小时 月内日期 月 周内日期 年(可选字段) 特殊字符Cron 触发器利用一系列特殊字符,如下所示: 反斜线(/)字符表 ...

  3. asd短片数篇

    黄乙己 黄乙己是站着AK而正常的唯一的人.他身材挺高大:蜡黄脸色,眼角间时常夹着些饼干屑:一副黑色的眼镜.虽然挺正常,可是他有良好的饮食习惯,似乎十多个月都是吃的牛奶泡饭,也没有洗饭盒.他对人说话,总 ...

  4. flagr 数据库配置

    flagr 是一个很不错的特性开关.a/b 测试服务,默认使用的是sqlite 数据库,但是我们可以通过配置,使用不同的数据库 sqlite.mysql.postrgresql.json_file.j ...

  5. idea 编译 内存不足

  6. linux下突破10万高并发的nginx性能优化经验

    一.这里的优化主要是指对nginx的配置优化,一般来说nginx配置文件中对优化比较有作用的主要有以下几项:1)nginx进程数,建议按照cpu数目来指定,一般跟cpu核数相同或为它的倍数.worke ...

  7. Linux内核中常用的数据结构和算法(转)

    知乎链接:https://zhuanlan.zhihu.com/p/58087261 Linux内核代码中广泛使用了数据结构和算法,其中最常用的两个是链表和红黑树. 链表 Linux内核代码大量使用了 ...

  8. 【转】Syncthing – 数据同步利器---自己的网盘,详细安装配置指南,内网使用,发现服务器配置

    Syncthing – 数据同步利器---自己的网盘,详细安装配置指南,内网使用,发现服务器配置 原贴:https://www.cnblogs.com/jackadam/p/8568833.html ...

  9. 开始转变方向,学习Linux——《Linux就该这么学》

    三十而立,四十不惑. 我呢,未立将不惑. 苦恼之余,决定拓展就业范围,正式学习Linux,准备考取RHCE证书. 考证需要报名培训机构,这是一个明智的选择,毕竟中国人善于考试,善于钻研考试. 联系培训 ...

  10. 程序员装X指南

      一.准备工作“工欲善其事必先利其器.” 1.电脑不一定要配置高,但是双屏是必须的,越大越好,能一个横屏一个竖屏更好.一个用来查资料,一个用来写代码 .总之要显得信息量很大,效率很高. 2.椅子不一 ...