zoukankan      html  css  js  c++  java
  • 装饰器

    1、装饰器介绍与简单实现

    1.1、什么是装饰器

    ​ 器:指的是具备某一个功能的工具

    ​ 装饰:指的是为被装饰对象添加新功能

    ​ 装饰器就是用来为被装饰对象添加新功能的工具。

    ​ 注意:装饰器本身可以是任意可调用对象,被装饰的对象也可以是任意可调用对象。

    1.2、为何要用装饰器

    开放封闭原则:封闭指的是对修改封闭,对扩展开放。

    ​ 装饰器的实现必须遵循两大原则:

    ​ 1、不修改被装饰对象的源代码。

    ​ 2、不修改被装饰对象的调用方式。

    1.3、怎么用装饰器

    # 给index函数添加上一个计时功能
    import time
    def index():
        print("welcome to index page")
        time.sleep(3)
    
    def outter(func):   # func=最原始的那个index的内存地址
        def wrapper():
            start = time.time()
            func()
            stop = time.time()
            print("run time is %s"%(stop - start))
        return wrapper
    
    index = outter(index)  # index = outter(最原始的那个index的内存地址) # index = wrapper的内存地址
    index()  # wrapper的内存地址()
    

    2、装饰器升级版

    ​ 考虑到某些函数会有返回值和参数,而上面的简易版装饰器并不能实现这两个条件,所以:

    import time
    def home(name):
        print("Welcome %s to home page"%name)
        time.sleep(2)
        return 123
    
    def timmer(func):  # func=最原始的那个index的内存地址
        def wrapper(*args,**kwargs):  # args = ("egon",) kwargs = {}
            start = time.time()
            res = func(*args,**kwargs)  # 最原始的那个home函数的内存地址("egon")
            stop = time.time()
            print("run time is %s"%(stop - start))
            return res
        return wrapper
    
    home = timmer(home)  # home=timmer(最原始那个home函数的内地址)  # home=wrapper函数的内地址
    res = home("egon")  # res=wrapper函数的内存地址('egon')
    print(res)
    

    3、装饰器的语法糖

    ​ 装饰器的语法糖:在被装饰对象正上方单独一行写@装饰器的名字。

    ​ 运行原理:

    ​ python解释器一旦运行到@装饰器的名字,就会调用装饰器,然后将被装饰函数的内存地址当作参数传给装饰器,最后将装饰器调用的结果赋值给原函数名。

    import time
    
    def timmer(func):  # func=最原始的那个index的内存地址
        def wrapper(*args,**kwargs):  # args = ("egon",) kwargs = {}
            start = time.time()
            res = func(*args,**kwargs)  # 最原始的那个home函数的内存地址("egon")
            stop = time.time()
            print("run time is %s"%(stop - start))
            return res
        return wrapper
    
    # 注意:语法糖要写在装饰器的后面,被装饰对象的前面
    @timmer  # index = timmer(index)
    def home(name):
        print("Welcome %s to home page"%name)
        time.sleep(2)
        return 123
    
    res = home("egon")  # res=wrapper函数的内地址('egon')
    print(res)
    

    装饰器模板

    def outter(func):
        def wrapper(*args,**kwargs):
            # 在调用函数前加功能
            res = func(*args,**kwargs)
            # 在调用函数后加功能
            return res
        return wrapper
    @outter
    def f():
        pass
    f()
    

    4、小练习:登录认证功能装饰器

    def auth(func):
        def wrapper(*args,**kwargs):
            inp_user = input('please input your username: ').strip()
            inp_pwd = input('please input your password: ').strip()
            if inp_user == 'egon' and inp_pwd == '123':
                print('login successfull')
                res=func(*args,**kwargs) # 调用最原始的那个/也就是被装饰的那个函数
                return res
            else:
                print('username or password error')
        return wrapper
    
    @auth # index=auth(index) #index=wrapper
    def index():
        print('welcome to index page')
        time.sleep(3)
    index() #wrapper()
    

    5、叠加多个装饰器

    # 解释@语法的时候是自下而上运行
    # 而执行装饰器内的那个wrapper函数时的是自上而下
    import time
    def outter1(func1):  # func1=wrapper2
        print('outter1')
        def wrapper1(*args,**kwargs):
            print('wrapper1')
            res1=func1(*args,**kwargs)  # res1=wrapper2(*args,**kwargs)
            return res1
        return wrapper1
    
    def outter2(func2):  # func2=最原始的那个index的内存地址
        print('outter2')
        def wrapper2(*args,**kwargs):
            print('wrapper2')
            res2=func2(*args,**kwargs)
            return res2
        return wrapper2
    
    @outter1  # index=outter1(wrapper2) #index=wrapper1
    @outter2  # outter2(最原始的那个index的内存地址) ===> wrapper2
    def index():
        print('welcome to index page')
        time.sleep(3)
    index()  #wrapper1()
    
    '''
    运行结果如下:
    outter2
    outter1
    wrapper1
    wrapper2
    '''
    

    6、有参装饰器

    import time
    
    current_user={'username':None}
    
    def login(engine='file'): #engine='mysql'
        def auth(func): #func=最原始那个index的内存地址
            def wrapper(*args,**kwargs):
                if current_user['username']:
                    print('已经登录过了,无需再次登陆')
                    res=func(*args,**kwargs)
                    return res
    
                if engine == 'file':
                    inp_user = input('please input your username: ').strip()
                    inp_pwd = input('please input your password: ').strip()
                    if inp_user == 'egon' and inp_pwd == '123':
                        print('login successfull')
                        current_user['username']=inp_user # 在登陆成功之后立刻记录登录状态
                        res=func(*args,**kwargs) # res=最原始那个index的内存地址(*args,**kwargs)
                        return res
                    else:
                        print('username or password error')
                elif engine == 'mysql':
                    print('基于mysql的认证机制')
                elif engine == 'ldap':
                    print('基于ldap的认证机制')
                else:
                    print('无法识别的认证源')
            return wrapper
        return auth
    
    @login('file')  #@auth # index=auth(最原始那个index的内存地址) #index=wrapper
    def index():
        print('welcome to index page')
        time.sleep(3)
    
    @login('file')
    def home(name):
        print('welcome %s to home page' %name)
        time.sleep(2)
        return 123
    
    
    index() #wrapper()
    res=home('egon')
    print(res)
    

    有参装饰器模板

    def outter1(x,y,z):
        def outter2(func):
            def wrapper(*args,**kwargs):
                res=func(*args,**kwargs)
                return res
            return wrapper
        return outter2
    

    无参装饰器模板

    def outter(func):
        def wrapper(*args,**kwargs):
            res=func(*args,**kwargs)
            return res
        return wrapper
    

    7、from functools import wraps

    ​ 在使用装饰器时,被装饰好的函数在调用时实际上是在调用wrapper函数,而wrapper函数并不具有原函数的所有名称空间内的值,比如函数名(func.name_)、函数功能(func.doc_)等,这时后就要用到from functools import wraps。

    #wraps装饰器应该加到装饰器最内层的函数上
    from functools import wraps
    import time
    
    def deco(func):
        @wraps(func)
        def wrapper(*args, **kwargs):
            res = func(*args, **kwargs)
            return res
        # wrapper.__name__=func.__name__
        # wrapper.__doc__=func.__doc__
        return wrapper
    
    @deco #index=deco(index) #index=wrapper函数的内存地址
    def index():
        """
        index 功能
        """
        print('welcome to index page')
        time.sleep(3)
        
    print(index.__name__)
    print(index.__doc__)
    
  • 相关阅读:
    選定下拉列表的方法
    JAVA学习路线
    封装
    重写(Overriding)
    java内存泄露
    学习实例.文章管理.文章类.Article.java
    方法重载
    学习实例.文章管理.分页查询类.Page.java
    学习实例.文章管理.目标与经验总结
    Java多线程1
  • 原文地址:https://www.cnblogs.com/chenwang-23/p/10968448.html
Copyright © 2011-2022 走看看