python装饰器中的指针

下面将给出两段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)

这一步而已,使程序看起来更加简洁。

你可能感兴趣的:(python装饰器中的指针)