Python装饰器——初识

作者: 大阿拉伯人 | 来源:发表于2016-07-17 11:07 被阅读719次

    上次讲了Python的闭包,今天来讲一下闭包的应用——装饰器

    1. 装饰器是什么

    什么叫装饰器?顾名思义,它是一个用来装饰的东西。用来装饰谁呢?用来装饰函数。由于装饰器器自身也是一个函数,所以,一句话来说,装饰器是一个用来装饰函数的函数。感觉有点拗口,那么,在我的理解,装饰器只是个辅助函数,有没有它并不影响被装饰函数的运行。下面听我娓娓道来。

    2.函数的表达

    在讲之前,我们先来看一下函数,有这样一个函数:

    def func():
        print "running func"
    

    我们都知道,运行func()将会输出:

    >> running func
    

    然而,运行func,则输出:

    >> <function func at 0x7f70236215f0>
    

    所以我们看到在函数名之后加了括号会进入函数内部运行,而不加括号只是代表了一个函数对象(Python中一切皆对象,函数也是),记住这点可以帮我们更好的理解装饰器。

    3.场景

    现在我有一堆函数例如:

    def func1():
        print "running func1"
    
    def func2():
        print "running func2"
    
    def func3():
        print "running func3"
    

    不过现在,为了证明这些函数是我写的,都要打印一句话。那么,我可能会这样写:

    def func1():
        print "running func1"
        print "强哥好帅!"
    

    其他函数也是如此。然而,到以后函数多了的时候,你会感觉这样写很是麻烦,而且还更改了原函数的业务逻辑,非常不好。这时候,便可以用闭包来实现。

    4.用闭包简化

    改成闭包的形式如下:

    #把此函数写在最上面,因为`Python`解释器自上而下解释
    def shoe_me(func):
        def wrapper():
            print "强哥好帅!"
            func()
        return wrapper
    
    #原函数
    def func1():
        print "running func1"
    
    func1 = show_me(func1)
    
    if __name__ == '__main__':
        func1()
    

    运行后的结果如图:

    闭包实现
    注意:当程序运行到func1 = show_me(func1)时并没有立即执行函数,只是定义了一个新函数。show_me(func1)其实等价于如下:
    def wrapper():
        print "强哥好帅!"
        func1()
    

    所以,只有当func1()时,才真正运行。这正好对应了第二部分中函数执行的原理。

    5.用装饰器进一步简化

    以上需求的实现还是有一点繁琐,Python大发给我们提供了@语法糖,来看一下怎么使用:

    @shoe_me
    def func2():
        print "running func2"
    

    运行程序func2(),输出如下图:

    装饰器实现
    如程序所示,@shoe_me和原来闭包实现func2 = show_me(func2)是完全等价的,这样便能在不改变原函数的情况下添加装饰,增加功能。

    显然,上面的例子除了装B,确实没有其他功能。那装饰器到底有什么实际功能呢?如函数日志分析(对上面的装饰器改进),登陆访问(Django中的@login_required),权限验证,缓存设置,记数器等,下一节将继续介绍装饰器的高级用法。

    6.后记

    略啰嗦,最后一部分才引入装饰器。。。

    相关文章

      网友评论

      • 晨潇凝暮:就喜欢作者这种简单粗暴的讲解
      • 9c68e84e9034:哥们讲的不错,就喜欢简单,粗暴,不啰嗦
        大阿拉伯人:@虎旦 谢谢,喜欢就好
      • 60288baebcc3:不错 符合python的简洁易懂 笔记不错学习下 楼主用的什么插件repl 还是else 我安装了repl 执行当前文件 不顶用
        60288baebcc3:我再找找 input问题的解决方法 谢谢
        大阿拉伯人:@ColinFeng 我好像没有用什么插件,你试一下文件的目录不要有中文,再ctrl+b运行一下
      • minningl:讲的好清晰
        大阿拉伯人:@边城矢梦lmy 谢谢,喜欢就好

      本文标题:Python装饰器——初识

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