Python 装饰器(Decorator)
装饰模式有很多经典的使用场景,例如插入日志、性能测试、事务处理等等,有了装饰器,就可以提取大量函数中与本身功能无关的类似代码,从而达到代码重用的目的。下面就一步步看看Python中的装饰器。
装饰器本身是一个Python函数,他可以让其他函数在不需要做任何代码变动的前提下增加额外功能,装饰器的返回值也是一个额外的(函数)对象。
什么时候使用装饰器?
一般在开发过程中,要遵循开放封闭原则,虽然在这个原则是用的面向对象开发,但是也适用于函数式编程,简单来说,它规定已经实现的功能代码不允许被修改,但可以被扩展,这时就要使用装饰器了。
使用装饰器来修饰函数
- 不改变原有的程序,并且可以添加新的功能
- 可以提高程序的可重复利用性,并增加了程序的可读性。
要了解装饰器,需要了解一些概念
作用域
在python中,作用域分为两种:全局作用域和局部作用域。
全局作用域是定义在文件级别的变量,函数名。而局部作用域,则是定义函数内部。
关于作用域,我们要理解两点:
a.在全局不能访问到局部定义的变量
b.在局部能够访问到全局定义的变量,但是不能修改全局定义的变量(当然有方法可以修改)
关于作用域的问题,只需要记住两点就行:
全局变量能够被文件任何地方引用,但修改只能在全局进行操作;如果局部没有找到所需的变量,就会往外进行查找,没有找到就会报错。
闭包
装饰器
装饰器:外部函数传入被装饰函数名,内部函数返回装饰函数名。
特点:1.不修改被装饰函数的调用方式 2.不修改被装饰函数的源代码
实例:
# FileName : PyDecorator.py # Author : Adil # DateTime : 2018/4/19 19:04 # SoftWare : PyCharm ''' 装饰器 ''' def outer(some_func): # 将foo函数(地址)赋值给some_func def inner(): print("before some_func") ret = some_func() # 这里调用foo 返回 1 print(ret) return ret + 1 # 返回 ret + 1 = 1 + 1 = 2 return inner # 返回 inner 函数(地址) def foo(): return 1 decorated = outer(foo) # 将函数 foo 作为参数传给 outer ,返回 inner 函数对象(函数地址)。 print(decorated) # 一个替代版函数,函数地址 print(decorated()) # 返回 ret + 1 = 1 + 1 = 2
函数装饰器 @ 符号的应用
Python2.4支持使用标识符@将装饰器应用在函数上,只需要在函数的定义前加上@和装饰器的名称。
@符号是装饰器的语法糖,在定义函数的时候使用,避免再一次赋值操作;
@outer # 替代后后面的操作 decorated = outer(foo) # 将函数 foo 作为参数传给 outer ,返回 inner 函数对象(函数地址)。 def foo(): print("foo") return 1
在上面的例子里我们是将原本的方法用装饰后的方法代替:
def outer(some_func): # 将foo函数(地址)赋值给some_func def inner(): print("before some_func") ret = some_func() # 这里调用foo 返回 1 print(ret) return ret + 1 # 返回 ret + 1 = 1 + 1 = 2 return inner # 返回 inner 函数(地址) @outer def foo(): print("foo") return 1 foo() print(foo()) # 执行结果 # before some_func # foo # 1 # before some_func # foo # 1 # 2
被装饰的函数有参数
def decoratorF(funa): def middle(param): print(param,end=',') funa(param) return middle @decoratorF def testF(param): print("我是 Yang") testF("Hello") # 执行结果 # Hello,我是 Yang
更通用的装饰器
有了这招新的技能,我们随随便便就可以写一个能够记录下传递给函数参数的装饰器了。先来个简单地把日志输出到界面的例子:
含多个参数的装饰器
def testL(funa): def inner(*args,**kwargs): print("paramers are : %s,%s" %(args,kwargs)) return funa(*args, **kwargs) return inner @testL def testf(x,y,**k): print("testf") testf('a','b',z='c',t= 'x') # 打印结果 # paramers are : ('a', 'b'),{'z': 'c', 't': 'x'} # testf
一个函数被多个装饰器装饰,执行顺序从下往上。
def timer(funt): def get_time(*args,**kwargs): start_time = time.time() funt(*args, **kwargs) stop_time = time.time() print('execute time is %s'%(stop_time-start_time)) return get_time def testL(funa): def inner(*args,**kwargs): print("paramers are : %s,%s" %(args,kwargs)) return funa(*args, **kwargs) return inner @timer @testL def testf(x,y,**k): print("testf") testf('a','b',z='c',t= 'x') # 打印结果 # paramers are : ('a', 'b'),{'z': 'c', 't': 'x'} # testf # execute time is 0.0
类装饰器
装饰器不仅可以是函数,还可以是类,相比函数装饰器,类装饰器具有灵活度大、高内聚、封装性等优点。
使用类装饰器主要依靠类的__call__方法,当使用 @ 形式将装饰器附加到函数上时,就会调用此方法。
装饰器函数其实是这样一个接口约束,它必须接受一个callable对象作为参数,然后返回一个callable对象。
在Python中一般callable对象都是函数,但也有例外。只要某个对象重写了 call() 方法,那么这个对象就是callable的。
class ClassTest(object): def __init__(self,func): self._func =func def __call__(self): print('class decorator running') self._func() print('class decorator ending') @ClassTest # fun_Test = ClassTest(fun_Test) def fun_Test(): print('fun_Test') fun_Test() # ClassTest(fun_Test)() # 执行结果 # class decorator running # fun_Test # class decorator ending class TestClass(object): def __init__(self): pass def __call__(self, func): def _call(*args,**kwargs): print('class decorator running') func(*args,**kwargs) return func(*args,**kwargs) return _call class Bar(object): @TestClass() def bar(self,test,ids): # bar = TestClass()(bar) print('bar') Bar().bar('Yang','123') # 执行结果 # class decorator running # bar # bar
class Fee(object): def __init__(self,func): self._func = func def __call__(self, name): print 'hello', self._func(name) @Fee def foo(gold): print gold, print 'lxshen' foo('大家好') 输出: hello 大家好 lxshen 分析: #1.当用Fee来装作装饰器对foo函数进行装饰的时候,首先会创建Fee的实例对象 # 并且会把foo这个函数名当做参数传递到__init__方法中 # 即在__init__方法中的func变量指向了foo函数体 #2. foo函数相当于指向了用Fee创建出来的实例对象 # #3. 当在使用foo()进行调用时,就相当于让这个对象(),因此会调用这个对象的__call__方法 # #4. 为了能够在__call__方法中调用原来foo指向的函数体,所以在__init__方法中就需要一个实例属性来保存这个函数体的引用 # 所以才有了self.__func = func这句代码,从而在调用__call__方法中能够调用到foo之前的函数体
Python内置装饰器
在Python中有三个内置的装饰器,都是跟class相关的:staticmethod、classmethod 和property。
- staticmethod 是类静态方法,其跟成员方法的区别是没有 self 参数,并且可以在类不进行实例化的情况下调用
- classmethod 与成员方法的区别在于所接收的第一个参数不是 self (类实例的指针),而是cls(当前类的具体类型)
- property 是属性的意思,表示可以通过通过类实例直接访问的信息