多应用+插件架构,代码干净,二开方便,首家独创一键云编译技术,文档视频完善,免费商用码云13.8K 广告
[TOC] 参考链接 * [银角大王讲装饰器](http://www.cnblogs.com/wupeiqi/articles/4980620.html#undefined) * [Python 迭代器/生成器/装饰器](https://www.jianshu.com/p/c77b71a690db) ## **引子** ### 开放封闭原则 在一个已经正常运行的项目中,对项目功能的增加,需要遵循开放封闭原则. 简单来说,它规定已经实现的功能代码不允许被修改,但可以被扩展,即: * 封闭:已实现的功能代码块不应该被修改 * 开放:对现有功能的扩展开放 ### 不改变调用原则 对已经功能的增改,不能修改原模块(函数)的调用方式,因为一个模块(函数)可能已经在N个地方使用了,如果改变了调用方式,就需要修改N个地方 ## **装饰器** 装饰器的作用就是**在不修改原函数的情况下添加新功能**,相当于高阶函数(将函数作为参数传入)和嵌套函数的结合. ### 装饰器基本格式: ```python def 装饰器函数(传入的函数): def 执行的嵌套函数(传入函数的参数): 装饰器语句 ... return 传入的函数(传入函数的参数) ... 装饰器语句 return 返回的嵌套函数 @装饰器函数 def 原函数 原函数模块... ``` ### 无参数的装饰器 ```python def w1(func): def inner(): # 验证1 # 验证2 # 验证3 return func() return inner @w1 def f1(): print 'f1' ``` ### 参数格式已知的装饰器 ``` def w1(func): def inner(arg1,arg2,arg3): # 验证1 # 验证2 # 验证3 return func(arg1,arg2,arg3) return inner @w1 def f1(arg1,arg2,arg3): print 'f1' ``` ### 参数格式未知的装饰器 ``` def w1(func): def inner(*args,**kwargs): # 验证1 # 验证2 # 验证3 return func(*args,**kwargs) return inner @w1 def f1(arg1,arg2,arg3): print 'f1' ``` ### 多个装饰器装饰一个函数 ``` def w1(func): def inner(*args,**kwargs): # 验证1 # 验证2 # 验证3 return func(*args,**kwargs) return inner def w2(func): def inner(*args,**kwargs): # 验证1 # 验证2 # 验证3 return func(*args,**kwargs) return inner @w1 @w2 def f1(arg1,arg2,arg3): print 'f1' ``` ## 装饰器流程详解 ### 1.简单代码 以上面的无参数的装饰器来做解释 ``` def w1(func): def inner(): # 验证代码1 return func() return inner @w1 def f1(): print 'f1' ``` ### 2.代码执行步骤 当写完这段代码后(函数未被执行、未被执行、未被执行),python解释器就会从上到下解释代码,步骤如下: 1. `def w1(func): ` ==>将w1函数加载到内存 2. `@w1 ` 没错,从表面上看解释器仅仅会解释这两句代码,因为函数在没有被调用之前其内部代码不会被执行。 从表面上看解释器着实会执行这两句,但是 @w1 这一句代码里却有大文章,@函数名 是python的一种语法糖。 ### 3.将原函数作为参数传给新函数 如上例@w1内部会执行一下操作: **执行w1函数**,并将 @w1 下面的 函数 作为w1函数的参数,即:@w1 等价于 w1(f1) 所以,内部就会去执行: ``` def inner: #验证 return f1() # func是参数,此时 func 等于 f1 return inner # 返回的 inner,inner代表的是函数,非执行函数 ``` >其实就是将原来的 f1 函数塞进另外一个函数中 ### 4.新函数执行完毕后再将原函数return返回 将执行完的 w1 函数**返回值赋值**给@w1下面的函数的函数名 ``` w1函数的返回值是: def inner: #验证 return 原来f1() # 此处的 f1 表示原来的f1函数 然后,将此返回值再重新赋值给 f1,即: 新f1 = def inner: #验证 return 原来f1() ``` >所以,以后业务部门想要执行 f1 函数时,就会执行 新f1 函数,在 新f1 函数内部先执行验证,再执行原来的f1函数,然后将 原来f1 函数的返回值 返回给了业务调用者。 如此一来, 即执行了验证的功能,又执行了原来f1函数的内容,并将原f1函数返回值 返回给业务调用着