zoukankan      html  css  js  c++  java
  • Python基础15_装饰器

     一. 装饰器
        开闭原则: 软件设计的原则之一, 又称为开放封闭原则. 
        开放: 对功能扩展开放
        封闭: 对修改代码封闭
        装饰器: 在目标函数前或后插入一段新的代码, 不改变目标函数的代码
        可以给目标函数传参, 拿到目标函数的返回值
        python里面的动态代理
        在不破坏目标函数和目标函数调用的基础上给函数添加新的功能
        通用语法:
            def wrapper(fn):                        # fn 是目标函数
                def inner(*args, **kwargs):            # 聚合    给目标函数传参
                    """在目标函数之前进行操作"""
                    ret = fn(*args, **kwargs)        # 打散    执行目标函数, 接收目标函数的返回值"xxx"
                    """在目标函数之后进行操作"""
                    return ret                        # 返回目标函数的返回值"xxx", 保证函数正常结束
                return inner                        # 返回inner的返回值"xxx"
            @wrapper                                # 语法糖 装饰器中特有 相当于 target_func = wrapper(target_func)
            def target_func()
                pass
                return "xxx"
            # target_func = wrapper(target_func)
            ret = target_func()                        # 执行inner, 并接收inner的返回值"xxx"
            print(ret)                                # 最终打印的是目标函数的返回值"xxx"
    二. 带参数的装饰器
        
        语法:
        def wrapper_out(形参):                    
            def wrapper(fn):
                def inner(*args, **kwargs):
                    """在目标函数之前进行操作"""
                    ret = fn(*args, **kwargs)
                    """在目标函数之后进行操作"""
                    return ret
                return inner
            return wrapper
        @wrapper_out(实参)
        def func():
            pass
        func()
    三. 多个装饰器装饰同一个函数
        def wrapper1(fn):
            def inner(*args, **kwargs):
                print(111111)
                ret = fn(*args, **kwargs)
                print(222222)
                return ret
            return inner
        def wrapper2(fn):
            def inner(*args, **kwargs):
                print(333333)
                ret = fn(*args, **kwargs)
                print(444444)
                return ret
                return inner
        @wrapper1
        @wrapper2
        def target_func():
            print("我是target_func")
            return "target_func"
        ret = target_func()
        print(ret)
        # 打印结果
            111111
            333333
            我是target_func
            444444
            222222
            target_func
        执行顺序: 首先@wrapper2装饰起来, 然后获取到一个新函数是wrapper2中的inner, 然后执行@wrapper1, 这个时候, wrapper1装饰的就是wrapper2中的inner了, 所以执行顺序就像: 
        外层装饰器前 -- 内层装饰器前 -- 目标 -- 内层装饰器后 -- 外层装饰器后 -- 目标函数返回值

  • 相关阅读:
    openresty开发系列35--openresty执行流程之5内容content阶段
    openresty开发系列34--openresty执行流程之4访问阶段
    openresty开发系列33--openresty执行流程之3重写rewrite和重定向
    [转] Dangers of using dlsym() with RTLD_NEXT
    fork failed because of Out Of Memory
    gdb 拾穗
    原子变量的性能问题
    blktrace + blkparse + btt 分析IO
    [转] 利用BLKTRACE分析IO性能
    使用perf + FlameGraph生成进程火焰图
  • 原文地址:https://www.cnblogs.com/guyannanfei/p/10121475.html
Copyright © 2011-2022 走看看