Python装饰器的调用过程

  在Python学习的过程中,装饰器是比较难理解的一个应用。本人也在学习期间也遇到很多坑,现将装饰器的基本调用过程总结一下。

  首先,装饰器用到了“闭包”,而“闭包”是学习装饰器的基础,所以在讲装饰器之前先将“闭包”的基本概念带大家一起回顾一下:

  现有如下函数:

def func():
    def func1():
        print('i\'m func1')
    return func1
if __name__ == '__main__':   f
= func()
  f()
  print('函数func1的内存地址为:',f)

  我们在函数func()内部定义了另外一个函数func1(),最终return的是func1的内存地址。 

  函数定义完后,在全局把func()赋值给变量f,此时f中拿到的是func1的内存地址,这时候你可以把f看成是func1,进行f()操作相当于调用func1(),所以上面代码的输出结果为:

 

  此时有些人可能会问:既然f相当于func1,那么为何不在全局中直接调用func1()呢?答案是不行~因为func1()是在全局的函数func()里面定义的,全局情况下只能调用func(),不可以直接调用func()里面的函数:

  这里我们可以看到:"func1 is not defined",因为在全局程序只能调用func(),是找不到func1()的,跟别提调用了。

  所以,我们要想在全局情况下调用“全局函数”内部定义的函数,就必须令该全局函数返回“内部函数”的内存地址,然后将该内存地址赋值给一个变量,通过调用这个变量来实现“全局调用内部函数”,而此时,这个“内部的函数”就称为“闭包”

  而上述例子中,函数func1就是一个闭包。

  理解了“闭包”的概念后我们再来看“装饰器”的调用过程:

  一:不带参数的装饰器

  

  如上图:这里我们先定义了一个装饰器Dec(),而Dec函数里面的outer函数就是一个闭包。当我们在函数func2定义前加上@Dec时,这个语句相当于:func2 = Dec(func2)。也就是说,我们在进行不带参数的装饰器的调用时,相当于把下面的函数名当做参数传给了@后面的函数,@Dec也就相当于执行了Dec(func2)。后面就好理解了:Dec()函数返回了outer函数的内存地址,下面的func2()其实就调用了“闭包”outer(),进行了outer()函数里面的操作。

  二:带参数的装饰器

  这里需要注意的是:如果要返回函数的话,带参数的装饰器就要写三层内嵌函数。

  带参数的装饰器的具体执行过程分为两步:首先执行Dec('QQ'),不管中间过程,Dec函数返回的是函数outer的内存地址,此时就变成了@outer,按照“不带参数的装饰器”的调用过程我们知道,此时outer将函数func2的名称当做是参数执行outer里面的函数inner()。另外我们还需要注意:现在inner里不仅有func2,还有Dec本身所携带的参数'QQ'。

  此外:打印出来的"i'm inner"是在判定if type == 'QQ'后直接执行的;而"i'm func2"是inner()函数执行“outer函数所带的参数”调用的结果,也就是说inner函数最后调用了“outer函数所带的参数”func并执行了它,换句话讲,inner就是一个闭包。

转载于:https://www.cnblogs.com/paulwhw/p/8683990.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值