python cookbook 学习系列(一) python中的装饰器
简介
装饰器本质上是一个Python函数,它可以让其他函数在不需要做任何代码变动的前提下增加额外功能,装饰器的返回值也是一个函数对象。它经常用于有切面需求的场景,比如:插入日志、性能测试、事务处理、缓存、权限校验等场景。装饰器是解决这类问题的绝佳设计,有了装饰器,我们就可以抽离出大量与函数功能本身无关的雷同代码并继续重用。概括的讲,装饰器的作用就是为已经存在的对象添加额外的功能,我们也称之为AOP(面向切面编程)
原理
实现装饰器的最主要的原因是python中一切皆为对象,我们会把方法看做一个对象包装起来
实现
首相,我们先来看一个简单的例子,有一个add方法:
def add(x, y): print('result is {}'.format(x + y))
现在有一个新的需求,我们想在这个方法执行完之后print一句话证明他执行成功了,同样我们又不想修改这个add方法,那我们可以新建一个方法:
def printres(func): func() print('Done') printres(add)
那这样是ok的。但是这样的话,我们每次都要将一个函数作为参数传递给printres函数。而且这种方式已经破坏了原有的代码逻辑结构,之前执行业务逻辑时,执行运行add(),但是现在不得不改成printres(add)。那么有没有更好的方式的呢?当然有,答案就是装饰器。我就直接贴代码:
# coding=utf-8 """ 学习装饰器 2016-08-20 by oldman """ from functools import wraps """ 先来看一个简单的装饰器 """ def decor1(func): @wraps(func) def wrapper(*args, **kwargs): func(*args, **kwargs) print('i am a decorate') return wrapper @decor1 def add(x, y): print('result is {}'.format(x + y)) # if __name__ =="__main__": # add(2,3) """ 再来看一个带参数的装饰器 """ def decor2(parm): """ parm为装饰器用到的参数 """ def _decor2(func): # func为被装饰的函数 @wraps(func) # wraps 装饰器会把func的元数据带过来 def wrapper(*args, **kwargs): if parm: func(*args, **kwargs) print('I am true') else: print('sorry, false') return wrapper return _decor2 @decor2(True) # 这里装饰器的参数为True def add1(x, y): print('result is {}'.format(x + y)) if __name__ == '__main__': add1(2, 3)
注意到functools.wraps,wraps本身也是一个装饰器,它能把原函数的元信息拷贝到装饰器函数中,这使得装饰器函数也有和原函数一样的元信息了,如果不在装饰器里使用@wrap的话,被装饰的函数的元信息会丢失。
总结
这是一个很好的解决方案,不得不感叹python的灵活,人生苦短,我用python。
参考文章:http://www.zhihu.com/question/26930016
学习交流群 :226704167