首页 > 编程 > Python > 正文

python实现装饰器、描述符

2020-02-22 23:20:37
字体:
来源:转载
供稿:网友

概要

本人python理论知识远达不到传授级别,写文章主要目的是自我总结,并不能照顾所有人,请见谅,文章结尾贴有相关链接可以作为补充

全文分为三个部分装饰器理论知识、装饰器应用、装饰器延申

装饰理基础:无参装饰器、有参装饰器、functiontools、装饰器链 装饰器进阶:property、staticmethod、classmethod源码分析(python代码实现)

装饰器基础

无参装饰器

'''假定有一个需求是:打印程序函数运行顺序此案例打印的结果为:  foo1 function is starting  foo2 function is starting'''from functools import wrapsdef NoParamDec(func):  #函数在被装饰器装时后,其函数属性也会改变,wraps作用就是保证被装饰函数属性不变  @wraps(func)  def warpper(*args, **kwargs):    print('{} function is starting'.format(func.__name__))    return func(*args, **kwargs)    return warpper#python黑魔法省略了NoParamDec=NoParamDec(foo1)@NoParamDecdef foo1():  foo2()@NoParamDecdef foo2():  passif __name__ == "__main__":  foo1()

有参装饰器

'''假定有一个需求是:检查函数参数的类型,只允许匹配正确的函数通过程序此案例打印结果为:('a', 'b', 'c')-----------------------分割线------------------------ERROS!!!!b must be <class 'str'> ERROS!!!!c must be <class 'str'> ('a', 2, ['b', 'd'])  '''from functools import wrapsfrom inspect import signaturedef typeAssert(*args, **kwargs):  deco_args = args  deco_kwargs = kwargs    def factor(func):    #python标准模块类,可以用来检查函数参数类型,只允许特定类型通过    sig = signature(func)    #将函数形式参数和规定类型进行绑定    check_bind_args = sig.bind_partial(*deco_args, **deco_kwargs).arguments        @wraps(func)    def wrapper(*args, **kwargs):      #将实际参数值和形式参数进行绑定      wrapper_bind_args = sig.bind(*args, **kwargs).arguments.items()      for name, obj in wrapper_bind_args:        #遍历判断是否实际参数值是规定参数的实例        if not isinstance(obj, check_bind_args[name]):          try:            raise TypeError('ERROS!!!!{arg} must be {obj} '.format(**{'arg': name, 'obj': check_bind_args[name]}))          except Exception as e:            print(e)      return func(*args, **kwargs)        return wrapper    return factor@typeAssert(str, str, str)def inspect_type(a, b, c):  return (a, b, c)if __name__ == "__main__":  print(inspect_type('a', 'b', 'c'))  print('{:-^50}'.format('分割线'))  print(inspect_type('a', 2, ['b', 'd']))

装饰器链

'''假定有一个需求是:输入类似代码:@makebold@makeitalicdef say():  return "Hello"输出:<b><i>Hello</i></b>'''from functools import wrapsdef html_deco(tag):  def decorator(fn):    @wraps(fn)    def wrapped(*args, **kwargs):      return '<{tag}>{fn_result}<{tag}>'.format(**{'tag': tag, 'fn_result': fn(*args, **kwargs)})        return wrapped    return decorator@html_deco('b')@html_deco('i')def greet(whom=''):  # 等价于 geet=html_deco('b')(html_deco('i)(geet))  return 'Hello' + (' ' + whom) if whom else ''if __name__ == "__main__":  print(greet('world')) # -> <b><i>Hello world</i></b>            
发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表