美文网首页
浅析decorator

浅析decorator

作者: Closears | 来源:发表于2015-06-15 19:32 被阅读70次

    从功能上来说,decorator是为了在代码运行期间动态增加代码功能的一种方案(即:我们要增强函数的功能,但是又不希望修改函数的定义);
    从实现上来说,decorator is a function that accepts a function as input and returns a new function as output.

    先上代码来举个栗子:


    example
    import time
    from functools import wraps
    
    def timethis(func):
        '''
        Decorator that reports the executions time.
        '''
        @wraps(func)
        def wrapper(*args, **kwargs):
            start = time.time()
            result = func(*args, **kwargs)
            end = time.time()
            print(func.__name__, end - start)
            return result
        return wrapper
    
    @timethis
    def countdown(n):
        '''
        Counts down
        '''
        while n > 0:
            n -= 1
    

    分析开始
    我们先来看原始函数:

    def countdown(n):
        '''
        Counts down
        '''
        while n > 0:
            n -= 1
    

    这个函数的功能很简单,就是把输入的n减到0,也没有输出,所以没什么卵用。接下来我们决定开始增强它的功能,让它有卵用!!!

    我决定给这个函数增加:
    1.打印函数名字的功能
    2.打印整个函数运行时间的功能

    这两个功能我已经写好了,我把它写在了名为timethis的这个函数里面。我现在就来改造原函数!代码如下:

    @timethis
    def countdown(n):
        '''
        Counts down
        '''
        while n > 0:
            n -= 1
    

    执行一下:print(countdown(100000))
    输出结果:

    countdown 0.014000892639160156
    None
    

    (不要在意这个None,之所以出现None是因为我们的coundown函数没有写返回值啊亲:P)

    你现在肯定想问我加了什么特技,居然只写了一个@timethis就实现了增强函数功能这么神奇的事情!

    讲解如下
    @这个符号在微博里表示“点名”,或者“提醒某某人”的意思,在这里也是一样的。
    @timethis写在了countdown函数定义的上面,就相当于countdown函数在微博上@了一下timethis,然后跟他说,“喂喂喂,内个谁,timethis啊,你帮我个忙啊。”

    那么countdown函数具体是如何告诉timethis自己的需求的呢?timethis怎么知道要帮什么忙呢?

    答案是:这些需求都已经写在了timethis函数的函数体里面,代码如下:

    def timethis(func):
        '''
        Decorator that reports the executions time.
        '''
        @wraps(func)  #这句代码先不要考虑,我们在本文章的最后给大家分析
        def wrapper(*args, **kwargs):
            start = time.time()  #获取系统时间
            result = func(*args, **kwargs)
            end = time.time()  #我们获取了两次系统时间,它们相减得到的值就是函数的运行时间
            print(func.__name__, end - start)
            return result
        return wrapper
    

    接下来我们一步步地还原这个函数的构建过程(注释我就不再写一遍了啊=_=):

    1.我们接收一个没卵用的函数,对它进行改造后,返回一个niubility的新函数。

    def timethis(func):  #func就是待改造的函数
        XXXXXXXXX   #对原函数func进行改造的代码
        return 一个增强后的新函数
    

    2.由于我们返回了一个新函数,所以理所当然地,我们要在函数体中定义这个新函数。(我们给这个增强后的函数取一个名字wrapper吧,这个名字取得还是挺形象的呢:P
    所以代码现在变成了如下的样子:

    def timethis(func):
        def wrapper(*args, **kwargs):  #参数表写成这个样子,表示我们可以将任意参数传入这个函数,在本文章末尾我会对此用法进行简要说明
            XXXXXXXXX  #增加的新功能的代码
            return func(*args, **kwargs)  #返回之前函数原有功能的执行结果(虽然我们要增加新功能,但是函数原有的功能也不能丢了呀,所以这段代码是必要的)
        return wrapper  #这句话返回了强大的新函数
    

    3.进一步地,函数变成了如下的样子:

    def timethis(func):
        def wrapper(*args, **kwargs):
            start = time.time()  #获取系统时间
            result = func(*args, **kwargs)
            end = time.time()  #我们获取了两次系统时间,它们相减得到的值就是函数的运行时间
            print(func.__name__, end - start)
            return result
        return wrapper
    

    到目前为止,我们基本上完成了decorator的编写和使用。
    我们谈一下前面遗留的几个重要问题:

    1.为什么要在wrapper函数的上面写@wraps(func) ?
    答:为了保留原函数的一些原始信息,或者说保留函数的metedata.

    当我们使用装饰器时(如下),

    @timethis
    def countdown(n):
        ...
    

    我们相当于执行了如下代码:

    def countdown(n):
        ...
    countdown = timethis(countdown)
    

    也就是说,我们返回了一个新函数后,还用旧函数的名字来使用它。那么问题来了,当我们通过这个函数来访问某些信息时(具体访问哪些信息我们接下来讨论),我们原本想访问旧函数的信息,可是由于被覆盖,我们只能访问到新函数的信息,这显然不是我们想要的结果。
    从另一个角度想,我们使用装饰器只是想增强原函数的功能,没想让新函数取而代之啊!我们需要的还是旧函数啊!
    好了,wrapper函数上面写的@wraps(func)就解决了这个问题。
    我们来访问一些信息,以便验证我们刚才说的话:

    condition 1:没写@wraps(func)时:

    执行如下代码:

    print(countdown.__name__)
    print(countdown.__doc__)
    

    输出如下:

    wrapper
    None
    

    condition 2:写了@wraps(func)以后:

    执行如下代码:

    print(countdown.__name__)
    print(countdown.__doc__)
    

    输出如下:

    countdown
    
        Counts down
    

    由此可见,@wraps(func)是非常必要的。补充一点,要想使用@wraps(func)还需要在代码的开头写上from functools import wraps这么一句。

    2.关于参数表(*args, **kwargs)
    * 代表可变参数,** 代表关键字参数。至于参数的名字,其实无所谓,不过通常我们使用args来命名可变参数,用kw或kwargs来命名关键字参数。
    这里我讲的不是很具体,不懂的话请百度“python可变参数”“python关键字参数”

    总结
    本文的内容就和本文的标题一样,只是对decorator的一个粗浅的介绍。
    还有很多有用的内容我们限于篇幅并没有提到,那么就请期待我的下一篇关于decorator的文章吧!

    相关文章

      网友评论

          本文标题:浅析decorator

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