概要
本人python理論知識遠(yuǎn)達(dá)不到傳授級別,寫文章主要目的是自我總結(jié),并不能照顧所有人,請見諒,文章結(jié)尾貼有相關(guān)鏈接可以作為補(bǔ)充
全文分為三個部分裝飾器理論知識、裝飾器應(yīng)用、裝飾器延申
裝飾理基礎(chǔ):無參裝飾器、有參裝飾器、functiontools、裝飾器鏈 裝飾器進(jìn)階:property、staticmethod、classmethod源碼分析(python代碼實(shí)現(xiàn))裝飾器基礎(chǔ)
無參裝飾器
'''假定有一個需求是:打印程序函數(shù)運(yùn)行順序此案例打印的結(jié)果為: foo1 function is starting foo2 function is starting'''from functools import wrapsdef NoParamDec(func): #函數(shù)在被裝飾器裝時后,其函數(shù)屬性也會改變,wraps作用就是保證被裝飾函數(shù)屬性不變 @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()有參裝飾器
'''假定有一個需求是:檢查函數(shù)參數(shù)的類型,只允許匹配正確的函數(shù)通過程序此案例打印結(jié)果為:('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標(biāo)準(zhǔn)模塊類,可以用來檢查函數(shù)參數(shù)類型,只允許特定類型通過 sig = signature(func) #將函數(shù)形式參數(shù)和規(guī)定類型進(jìn)行綁定 check_bind_args = sig.bind_partial(*deco_args, **deco_kwargs).arguments @wraps(func) def wrapper(*args, **kwargs): #將實(shí)際參數(shù)值和形式參數(shù)進(jìn)行綁定 wrapper_bind_args = sig.bind(*args, **kwargs).arguments.items() for name, obj in wrapper_bind_args: #遍歷判斷是否實(shí)際參數(shù)值是規(guī)定參數(shù)的實(shí)例 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>
新聞熱點(diǎn)
疑難解答
圖片精選