Python中的自省與反射
由於Python是一門強類型的動態解釋型語言,故我們在某些時候並不會知道(特別是與別人對接開發工作的時候)對象中具有的屬性與方法。
這個時候我們並不能直接通過 .或者查看底層的 __dict__ 方法來獲得該對象下的屬性與方法,我們需要使用一種更文明的方式來獲取該對象下的屬性與方法,故這種文明的方式被稱之為反射。
自省和反射是兩個比較專業化的術語,首先自省是獲取對象的能力,而反射是操縱對象的能力。
Python中使用delattr()和setattr()實現反射,而其他方法則屬於自省。
反射和自省常常組合使用!
Python中關於反射與自省的部分方法 | |
---|---|
常用方法 | |
dir() | 返回一個列表,存儲該對象下能被 . 出的所有屬性與方法。 |
hasattr() | 查看對象是否具有某種屬性或方法,返回True或者False。 |
getattr() | 獲取對象下的某一屬性或方法。如被獲取對象沒有相應的屬性或方法,則可以為其設置默認值。 |
setattr() | 設置對象下的某一屬性的值,通常我們不會在對象外部為其新增某一方法,而是在在對象的類中進行設置。 |
delattr() | 刪除對象中的某一屬性或方法。 |
其他的一些方法 | |
help() | 獲取對象的幫助信息,注意。沒有返回值!內部會調用print()進行打印操作。 |
issubclass() | 判斷一個類是不是另一個類的子類 |
isinstance() | 判斷一個對象是否是一個已知的類型 |
id() | 返回存儲對象的內存地址編號 |
callable() | 判斷對象是否可調用 |
注意:於一切皆對象的原則,我們不僅可以對實例對象進行自省與反射,也可以對類對象進行自省與反射。
操作演示
應用場景
# ==== 這樣做的好處是即使用戶輸入有誤,也不會拋出異常 ==== import sys class DownloadAndUpload(object): def __init__(self): self.val = sys.argv[1] self.select() def download(self): print("正在下載...") def upload(self): print("正在上傳...") def select(self): if hasattr(self,self.val): getattr(self,self.val)() else: print("沒有該方法") DownloadAndUpload()
擴展與後言:反射內部實現機制
其實我想了好一會要不要寫這個,內部實現機制。這一些內容應該放在雙下方法學完之後才應該講反射實現的內部機制。所以這裡提一嘴:
hasattr() : __getattribute__ 有則返回,無則捕捉異常返回False。
getattr() :這個應該是在描述符之後進行講解,實際上還是對__dict__進行操作。
setattr() : 調用內部__setattr__ 對 __dict__ 進行操作。Ps:實例對象調用時檢查其類及其父類,類對象調用時檢查其父類或者元類。
delattr() :調用__delattr__ 對 __dict__ 進行操作。Ps:實例對象調用時檢查其類及其父類,類對象調用時檢查其父類或者元類。
[火星人 ] Python自省及反射原理實例詳解已經有345次圍觀