下面将给出两段python代码,通过这段代码来解释python装饰器中的指针。在这之前,我们首先要了解的是python中函数的闭包,其实就是一句话,在python中的作用域有LEGB四种,而且在python中函数也是一种对象,也有自己的属性,其中有一个属性就叫做_enclosure_,这个属性里面就存储了内部函数对于外部函数的变量的引用地址。
第一段代码(这段代码使用了装饰器的syntactic sugar):
# coding=utf-8
def deco(fun):
def wraped_fun(x1, x2):
print('wrap code!')
fun(x1, x2)
return wraped_fun
@deco
def plus(x1, x2):
print(x1 + x2)
if __name__ == '__main__':
plus(2, 6)
这段代码中时使用了装饰器,而我们是要去了解装饰器的内部指针,所以现在我们来看另一段代码,它的功能和第一段代码是一模一样的,是装饰器的手动实现。
第二段代码(这段代码没有使用装饰器的syntactic sugar):
# coding=utf-8
def deco(fun):
def wraped_fun(x1, x2):
print('wrap code!')
fun(x1, x2)
return wraped_fun
def plus(x1, x2): # 这里没有@deco
print(x1 + x2)
if __name__ == '__main__':
plus = deco(plus) # 没有这一步
plus(2, 6)
第二段代码中plus这个名字指向的地址是plus这个函数在内存中的地址,但是当调用了deco这个函数后plus这个名字指向的地址是wraped_fun这个函数的内存地址,而原来plus的内存地址则存放在了wraped_fun的_enclosure_属性中,所以现在再调用plus(2,6)时,实际上是调用的wraped_fun函数,而在wraped_fun函数中又调用了原来的plus函数,达到了装饰的效果。
python装饰器语法糖实际上是替我们做了
plus = deco(plus)
这一步而已,使程序看起来更加简洁。
网友评论