一个例子解释python装饰器

作者: Kaijien | 来源:发表于2018-02-23 22:34 被阅读0次

    谈到python装饰器,一般都先提到函数式编程,但是其实运用python装饰器,和知不知道他是函数式编程没啥关系。

    例子

    def decorator(func):
        def wrapper(*args, **kwargs):
            print('我是装饰器呦')
            return  func(*args, **kwargs)
        return wrapper
        
    @decorator
    def test():
        print('我是test')
        
    test()
    

    输出如下:

    我是装饰器呦
    我是test
    

    其实就这么简单,完全可以把代码里的@理解为我们聊天中用的@,在调用某一个函数的时候@你想调用的装饰器函数,然后就可以在函数自身代码不变的情况下、增添一些具有普适性的功能。

    应用

    那么装饰器能做什么呢?emmm。。。最简单的一个应用,计算函数运行时间啊。

    import time
    
    def run_time(func):
        def wrapper(*args, **kw):
            t = time.time()
            ans = func(*args, **kw)
            t = time.time() - t
            return ans, t
        return wrapper
        
    @run_time
    def test():
        x=1
        for i in range(10 ** 6):
            x += 1
        return x
        
    test()
    

    输出如下:

    (1000001, 0.08555388450622559)
    

    曾几何时,我为了计算每种目标跟踪算法处理一帧图片的时间,写了多少重复代码,这不是优雅多了==!

    当然,装饰器的应用不止如此,它还可以传递参数,著名的用法比如web框架flask的路由配置,有兴趣可以阅读一下它的源码,很漂亮。

    所需基础

    为了彻底理解装饰器,有两个基础知识要巩固一下。一个是闭包函数,一个是可变参数。这里放上两个链接,个人觉得讲得比较好。
    https://www.liaoxuefeng.com/wiki/0014316089557264a6b348958f449949df42a6d3a2e542c000/001431752945034eb82ac80a3e64b9bb4929b16eeed1eb9000

    https://foofish.net/python-closure.html

    补充

    • 我们给上面的装饰器加一个参数:
    import time
    
    def run_time(text):
        def decorator(func):
            def wrapper(*args, **kwargs):
                t = time.time()
                ans = func(*args, **kwargs)
                t = time.time() - t
                if t>text:
                    return ans, '太慢'
                else:
                    return ans, '正常'    
            return wrapper
        return decorator
        
    @run_time(1)
    def test():
        x=1
        for i in range(10 ** 6):
            x += 1
        return x
        
    test( )
    

    相关文章

      网友评论

        本文标题:一个例子解释python装饰器

        本文链接:https://www.haomeiwen.com/subject/llsfxftx.html