一直在用别人写的装饰器,从来没有对其原理进行深入的探究。今天趁有点闲着的时间,把装饰器的原理好好看了一遍,做一下整理。
一、装饰器的基本原理
装饰器就是一个可以接受调用也可以返回调用的调用。装饰器本身是一个函数或方法,接受被装饰的函数作为其位置参数。装饰器通过使用该参数来执行某些操作,然后返回原始数据或其他一些调用。
先来一个基本的装饰器:
def decorater_by(func): func.__doc__="被装饰器重新装饰后的帮助文档" return func def add(x,y): return x+y add = decorater_by(add)
之后用help(app),查看打印结果。
Help on function add in module __main__:
add(x, y)
被装饰器重新装饰后的帮助文档
这个例子虽然很简单,但是有助于我们理解装饰的装饰过程:
1、将被装饰的函数名作为参数传递给装饰器
2、装饰器对传递进来的参数做某些操作
3、将处理后的数据作为返回值,重新赋值给原来的函数名。完成对被装饰函数的调用。
二、装饰器的语法
在python2.4之后,python加入了装饰器的特殊语法。定义完成装饰器之后,如果要应用装饰,就在被装饰的函数声明之前,添加一行代码,代码的格式为:@装饰器的名称。
还是以之前的例子为例:
def decorater_by(func): func.__doc__="被装饰器重新装饰后的帮助文档" return func @decorater_by def add(x,y): return x+y
print(add.__doc__)
打印的结果:被装饰器重新装饰后的帮助文档
这种语法和原始的方式在应用的方式上都是一样的。
你还可以通过多个@来应用多个装饰器,这就涉及到装饰器的应用顺序了。由于python是解释性语言,代码在执行时候是从上往下执行的。但是装饰器在应用的时候确实自下向上应用的,这看起来有些违反直觉。
好像没说明白,还是在举栗子吧。
def decorater_by(func): func.__doc__=func.__doc__+" 这是第一个装饰器 " return func def decorater_by_alse(func): func.__doc__=func.__doc__+" 这是第二个装饰器 " return func @decorater_by_alse @decorater_by def add(x,y): """我是函数本身 """ return x+y print(add.__doc__)
“”“”
我是函数本身 这是第一个装饰器 这是第二个装饰器 “”“”
用原始的方式去理解
add=decorater_by_alse(decorater_by(add))
三、装饰器的用途:
1、类型检查
2、保存帮助信息
3、用户验证
4、数据序列化
5、日志管理
。。。。
def decorater_by(func):
func.__doc__="被装饰器重新装饰后的帮助文档"
return func
def add(x,y):
return x+y
add = decorater_by(add)