python實(shí)現(xiàn)裝飾器、描述符
概要
本人python理論知識(shí)遠(yuǎn)達(dá)不到傳授級(jí)別,寫文章主要目的是自我總結(jié),并不能照顧所有人,請(qǐng)見(jiàn)諒,文章結(jié)尾貼有相關(guān)鏈接可以作為補(bǔ)充
全文分為三個(gè)部分裝飾器理論知識(shí)、裝飾器應(yīng)用、裝飾器延申
- 裝飾理基礎(chǔ):無(wú)參裝飾器、有參裝飾器、functiontools、裝飾器鏈
- 裝飾器進(jìn)階:property、staticmethod、classmethod源碼分析(python代碼實(shí)現(xiàn))
裝飾器基礎(chǔ)
無(wú)參裝飾器
'''
假定有一個(gè)需求是:打印程序函數(shù)運(yùn)行順序
此案例打印的結(jié)果為:
foo1 function is starting
foo2 function is starting
'''
from functools import wraps
def NoParamDec(func):
#函數(shù)在被裝飾器裝時(shí)后,其函數(shù)屬性也會(huì)改變,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)
@NoParamDec
def foo1():
foo2()
@NoParamDec
def foo2():
pass
if __name__ == "__main__":
foo1()
有參裝飾器
'''
假定有一個(gè)需求是:檢查函數(shù)參數(shù)的類型,只允許匹配正確的函數(shù)通過(guò)程序
此案例打印結(jié)果為:
('a', 'b', 'c')
-----------------------分割線------------------------
ERROS!!!!b must be <class 'str'>
ERROS!!!!c must be <class 'str'>
('a', 2, ['b', 'd'])
'''
from functools import wraps
from inspect import signature
def typeAssert(*args, **kwargs):
deco_args = args
deco_kwargs = kwargs
def factor(func):
#python標(biāo)準(zhǔn)模塊類,可以用來(lái)檢查函數(shù)參數(shù)類型,只允許特定類型通過(guò)
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']))
裝飾器鏈
'''
假定有一個(gè)需求是:
輸入類似代碼:
@makebold
@makeitalic
def say():
return "Hello"
輸出:
<b><i>Hello</i></b>
'''
from functools import wraps
def 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=''):
# 等價(jià)于 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>
裝飾器進(jìn)階
property 原理
通常,描述符是具有“綁定行為”的對(duì)象屬性,其屬性訪問(wèn)已經(jīng)被描述符協(xié)議中的方法覆蓋。這些方法是__get__()、__set__()和__delete__()。如果一個(gè)對(duì)象定義這些方法中的任何一個(gè),它被稱為一個(gè)描述符。如果對(duì)象定義__get__()和__set__(),則它被認(rèn)為是數(shù)據(jù)描述符。僅定義__get__()的描述器稱為非數(shù)據(jù)描述符(它們通常用于方法,但是其他用途也是可能的)。
屬性查找優(yōu)先級(jí)為:
- 類屬性
- 數(shù)據(jù)描述符
- 實(shí)例屬性
- 非數(shù)據(jù)描述符
- 默認(rèn)為_(kāi)_getattr__()
class Property(object):
'''
內(nèi)部property是用c實(shí)現(xiàn)的,這里用python模擬實(shí)現(xiàn)property功能
代碼參考官方doc文檔
'''
def __init__(self, fget=None, fset=None, fdel=None, doc=None):
self.fget = fget
self.fset = fset
self.fdel = fdel
self.__doc__ = doc
def __get__(self, obj, objtype=None):
if obj is None:
return self
if self.fget is None:
raise (AttributeError, "unreadable attribute")
print('self={},obj={},objtype={}'.format(self,obj,objtype))
return self.fget(obj)
def __set__(self, obj, value):
if self.fset is None:
raise (AttributeError, "can't set attribute")
self.fset(obj, value)
def __delete__(self, obj):
if self.fdel is None:
raise (AttributeError, "can't delete attribute")
self.fdel(obj)
def getter(self, fget):
return type(self)(fget, self.fset, self.fdel, self.__doc__)
def setter(self, fset):
return type(self)(self.fget, fset, self.fdel, self.__doc__)
def deleter(self, fdel):
return type(self)(self.fget, self.fset, fdel, self.__doc__)
class Student( object ):
@Property
def score( self ):
return self._score
@score.setter
def score( self, val ):
if not isinstance( val, int ):
raise ValueError( 'score must be an integer!' )
if val > 100 or val < 0:
raise ValueError( 'score must between 0 ~ 100!' )
self._score = val
if __name__ == "__main__":
s = Student()
s.score = 60
s.score
staticmethod 原理
@staticmethod means: when this method is called, we don't pass an instance of the class to it (as we normally do with methods). This means you can put a function inside a class but you can't access the instance of that class (this is useful when your method does not use the instance).
class StaticMethod(object):
"python代碼實(shí)現(xiàn)staticmethod原理"
def __init__(self, f):
self.f = f
def __get__(self, obj, objtype=None):
return self.f
class E(object):
#StaticMethod=StaticMethod(f)
@StaticMethod
def f( x):
return x
if __name__ == "__main__":
print(E.f('staticMethod Test'))
classmethod
@staticmethod means: when this method is called, we don't pass an instance of the class to it (as we normally do with methods). This means you can put a function inside a class but you can't access the instance of that class (this is useful when your method does not use the instance).
class ClassMethod(object):
"python代碼實(shí)現(xiàn)classmethod原理"
def __init__(self, f):
self.f = f
def __get__(self, obj, klass=None):
if klass is None:
klass = type(obj)
def newfunc(*args):
return self.f(klass, *args)
return newfunc
class E(object):
#ClassMethod=ClassMethod(f)
@ClassMethod
def f(cls,x):
return x
if __name__ == "__main__":
print(E().f('classMethod Test'))
相關(guān)文章
python遞歸調(diào)用中的坑:打印有值, 返回卻None
這篇文章主要介紹了python遞歸調(diào)用中的坑:打印有值, 返回卻None,本文通過(guò)問(wèn)題分析給出解決方法,對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-03-03
Python?Pygame實(shí)現(xiàn)可控制的煙花游戲
大家好,本篇文章主要講的是Python?Pygame實(shí)現(xiàn)可控制的煙花游戲,感興趣的同學(xué)趕快來(lái)看一看吧,對(duì)你有幫助的話記得收藏一下2022-01-01
基于Python編寫一個(gè)圖片識(shí)別系統(tǒng)
本項(xiàng)目將使用python3去識(shí)別圖片是否為色情圖片,會(huì)使用到PIL這個(gè)圖像處理庫(kù),并且編寫算法來(lái)劃分圖像的皮膚區(qū)域,感興趣的可以了解一下2022-06-06
在Python中使用filter去除列表中值為假及空字符串的例子
今天小編就為大家分享一篇在Python中使用filter去除列表中值為假及空字符串的例子,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-11-11
windows下python虛擬環(huán)境virtualenv安裝和使用詳解
這篇文章主要介紹了windows下python虛擬環(huán)境virtualenv安裝和使用詳解,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2019-07-07
用Python的Flask框架結(jié)合MySQL寫一個(gè)內(nèi)存監(jiān)控程序
這篇文章主要介紹了用Python的Flask框架結(jié)合MySQL些一個(gè)內(nèi)存監(jiān)控程序的例子,并且能將結(jié)果作簡(jiǎn)單的圖形化顯示,需要的朋友可以參考下2015-11-11
R語(yǔ)言屬性知識(shí)點(diǎn)總結(jié)及實(shí)例
在本篇文章里小編給大家整理了一篇關(guān)于R語(yǔ)言屬性知識(shí)點(diǎn)總結(jié)及實(shí)例內(nèi)容,有興趣的朋友們可以學(xué)習(xí)下。2021-03-03

