美文网首页
python装饰器

python装饰器

作者: StephenZhang01 | 来源:发表于2018-09-02 19:12 被阅读0次

    装饰器作用

            代码运行期间动态增加功能,而不用修改原函数(被装饰的函数)的方式。使用@语法,把decorator置于函数的定义处。


    装饰器原则

            1.不改变被装饰函数的源代码

            2.不改变被装饰函数的调用方式


    装饰器的划分

            装饰器可以划分为无参数装饰器与有参装饰器,带不带参数根据函数的需要自行设计。

    装饰器演化过程

            下面的代码是以无参数装饰器为例

    1)新函数里调用原来函数

    def old(x, y):

        print(x, y)

    def new(x, y):

        print("new function")

        old(x, y)

    new(3, 4)

    2)改进,将函数对象作为参数导入新函数中

    def old(x, y):

        print(x, y)

    def new(func, x, y):

        print("new function")

        old = func(x, y)

        return old

    new(old, 3, 4)

    3)继续改进,如果原函数需要任意个参数带入。则新函数需要定义可变位置参数与可变关键字参数。同时在调用原函数时,对输入的可变参数进行解构(参数符不符合需要在原函数中确认)。

    def old(x, y):

        print(x, y)

    def new(func, *args, **kwargs): 

        print("new function")

        old = func(*args, **kwargs)

        return old

    new(old, 3, 4)

    4)对新函数进行柯里化,逐步形成闭包结构

    def old(x, y):

        print(x, y)

    def new(func):

        def inner(*args, **kwargs):

            print("new function")

            old = func(*args, **kwargs) 

            return old

        return inner

    old = new(old)  # 后面old是原函数对象。通过新函数调用后返回inner对象给old标识符覆盖。inner函数封装着内存中的原函数的对象。 old = inner

    old(3, 4)

    5)改为装饰器

    def new(func):

        def inner(*args, **kwargs):

            print("new function")

            old = func(*args, **kwargs)

            return old

        return inner

    @new  #相当于old = new(old)或old=inner 

    def old(x, y):

        print(x, y)

    old(3, 4)  # 相当于 new(old)(3, 4) 或 inner(3, 4)


    装饰器练习

    1)将下面字典d进行扁平化输出,并输出其最大节点深度

    d = {'a':{'b':1, 'c':2}, 'd':{'e':3, 'f':{'g': 4}}}

    d2 = {}

    def outer(d2):

        def fix(func):

            def _fix(tmp = 0, tmp_index = "", *args, **kwargs):

                ret = func(*args, **kwargs)

                for i in d2:

                    if len(i.split(".")) > tmp:

                        tmp = len(i.split("."))

                        tmp_target = i

                return ret, tmp, tmp_target

            return _fix

        return fix

    @outer(d2)

    def new(dic=d, key=""):

        for k, v in dic.items():

            if isinstance(v, dict):

                new(dic=v, key=key+k+".")

            else:

                d2[key + k] = v

        return d2

    print(new())


    相关文章

      网友评论

          本文标题:python装饰器

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