为了账号安全,请及时绑定邮箱和手机立即绑定

装饰器问题归纳

标签:
Python

1.什么是装饰器?

复制代码

import timedef timmer(func):
    start = time.time()
    func()
    end = time.time()    print(end - start)    # return end - start@timmerdef foo():
    time.sleep(3)    print('hhh')

foo #1#执行结果:#hhh#3.0004751682281494

复制代码

 

以上例子我们通过一个timmer装饰器希望计算foo函数的运行时间

整个过程其实可以理解为:一个函数的嵌套

即将foo函数嵌套入了timmer函数,类似于timmer(foo)

 

思考:但是为什么#1位置调用foo函数时不是使用foo()而是foo呢???

我们试一下

复制代码

import timedef timmer(func):
    start = time.time()
    func()
    end = time.time()    print(end - start)    # return end - start@timmerdef foo():
    time.sleep(3)    print('hhh')

foo()# 执行结果:# hhh# Traceback (most recent call last):# 3.000311851501465#   File "E:/python/ftp_work/test.py", line 18, in <module>#     foo()# TypeError: 'NoneType' object is not callable

复制代码

错误显示:'NoneType' object is not callable

原因是python装饰器设计在这一点上其实是不友好的

foo()相当于timmer(foo)(),而timmer(foo)()并不是一个可以运行的函数,因为timmer函数没有一个可执行的返回值

我们做如下修改

复制代码

import timedef timmer(func):    def inner():
        start = time.time()
        func()
        end = time.time()        print(end - start)    return inner
@timmerdef foo():
    time.sleep(3)    print('hhh')

foo()# 执行结果:# hhh# 3.000600576400757

复制代码

得到了我们想要的结果

上面的修改给timmer函数的调用给一个inner返回值

所以执行timmer(foo)()时,就将timmer(foo)变为了一个可执行的函数

 

 

2.带有装饰器的函数有参数的情况

复制代码

import timedef timmer(func):    def inner(i):
        start = time.time()
        func(i)
        end = time.time()        print(end - start)    return inner
@timmerdef foo(i):
    time.sleep(3)    print(i)

foo('hhh')# 执行结果:# hhh# 3.000600576400757

复制代码

这里需要注意在foo函数的参数给装饰器传递时

并不是传递给了timmer函数,而是inner函数,类似于foo(i) == timmer(inner)(i) == inner(i)

 

3.装饰器带参数的情况

这种情况我们需要在装饰器外面再嵌套一个函数来专门接收装饰器传来的参数

 

复制代码

import timedef new_timmer(j):    print(j)    def timmer(func):        def inner(i):
            start = time.time()
            func(i)
            end = time.time()            print(end - start)        return inner    return timmer


@new_timmer('xxx')def foo(i):
    time.sleep(3)    print(i)

foo('hhh')# 执行结果:# xxx# hhh# 3.0002424716949463

复制代码

可以理解为foo(i) == new_timmer(j)()(i) == timmer(foo)(i) == inner(i)

 

 

 

参考:https://www.tuicool.com/articles/FBZvya

原文出处:https://www.cnblogs.com/Rongze-blog/p/9575285.html

点击查看更多内容
TA 点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
  • 推荐
  • 评论
  • 收藏
  • 共同学习,写下你的评论
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
微信客服

购课补贴
联系客服咨询优惠详情

帮助反馈 APP下载

慕课网APP
您的移动学习伙伴

公众号

扫描二维码
关注慕课网微信公众号

举报

0/150
提交
取消