Python一切皆物件(object),每個物件都可能有多個屬性(attribute)。 Python的屬性有一套統一的管理方案。
物件的屬性可能來自於其類別定義,稱為類別屬性(class attribute)。
類別屬性可能來自類別定義自身,也可能根據類別定義繼承來的。
一個物件的屬性也可能是該物件實例定義的,叫做物件屬性(object attribute)。
物件的屬性儲存在物件的__dict__屬性中。
__dict__為一個字典,鍵為屬性名,對應的值為屬性本身。我們看下面的類別和物件。
對應Java的反射中,來取得物件的屬性,如:
public class UserBean { private Integer id; private int age; private String name; private String address; } //类实例化 UserBean bean = new UserBean(); bean.setId(100); bean.setAddress("武汉"); //得到类对象 Class userCla = (Class) bean.getClass(); //得到类中的所有属性集合 Field[] fs = userCla.getDeclaredFields(); ......
class bird(object): feather = True class chicken(bird): fly = False def __init__(self, age): self.age = age summer = chicken(2) print(bird.__dict__) print(chicken.__dict__) print(summer.__dict__)
輸出:
{'__dict__':
{'fly': False, '__module__': '__main__', '__doc__': None, ' __init__':
{'age': 2}
第一行為bird類的屬性,如feather。
第二行為chicken類別的屬性,例如fly和__init__方法。
第三行為summer物件的屬性,也就是age。
有一些屬性,像是__doc__,不是由我們定義的,而是由Python自動產生。此外,bird類別也有父類,是object類別(正如我們的bird定義,class bird(object))。這個object類別是Python中所有類別的父類別。
也就是子類別的屬性,會覆寫父類別的屬性。
可以透過下面2中方法修改類別的屬性:
summer.__dict__['age'] = 3 print(summer.__dict__['age']) summer.age = 5 print(summer.age)
Python中的property
同一個物件的不同屬性之間可能存在依賴關係。當某個屬性被修改時,我們希望依賴該屬性的其他屬性也同時改變。這時,我們不能透過__dict__的方式來靜態的儲存屬性。 Python提供了多種即時產生屬性的方法。其中一種稱為特性(property)。
class bird(object): feather = True #extends bird class class chicken(bird): fly = False def __init__(self, age): self.age = age def getAdult(self): if self.age > 1.0: return True else: return False adult = property(getAdult) # property is built-in summer = chicken(2) print(summer.adult) summer.age = 0.5 print(summer.adult)
這裡的功能類似觸發器。在每次取得adult屬性的時候,會觸發getAdult的值。
特性使用內建函數property()來建立。 property()最多可以載入四個參數。前三個參數為函數,分別用於處理查詢特性、修改特性、刪除特性。最後一個參數為特性的文檔,可以為一個字串,起說明作用。
class num(object): def __init__(self, value): self.value = value print '<--init' def getNeg(self): print '<--getNeg' return self.value * -1 def setNeg(self, value): print '<--setNeg' self.value = (-1) * value def delNeg(self): print("value also deleted") del self.value neg = property(getNeg, setNeg, delNeg, "I'm negative") x = num(1.1) print(x.neg) x.neg = -22 print(x.value) print(num.neg.__doc__) del x.neg
整個過程之中,都沒有呼叫對應的幾個函數。
也就是說,neg這個屬性的創建,設置,刪除都透過property()註冊起來了。
Python特殊方法__getattr__ (這個常用)
我們可以用__getattr__(self, name)來查詢即時產生的屬性。
在pyhton中,物件屬性都是動態的,隨時可以根據需要新增或刪除屬性。
那麼getattr的作用就是,在產生這些屬性的時候,進行一層判斷處理操作。
例如:
class bird(object): feather = True class chicken(bird): fly = False def __init__(self, age): self.age = age def __getattr__(self, name): if name == 'adult': if self.age > 1.0: return True else: return False else: raise AttributeError(name) summer = chicken(2) print(summer.adult) summer.age = 0.5 print(summer.adult) print(summer.male)
每個特性需要有自己的處理函數,而__getattr__可以將所有的即時產生屬性放在同一個函數中處理。 __getattr__可以根據函數名區別處理不同的屬性。例如上面我們查詢屬性名male的時候,raise AttributeError。
(Python中還有一個__getattribute__特殊方法,用於查詢任意屬性。
__getattr__只能用來查詢不在__dict__系統中的屬性)
__setattr__(self, name, value)和__ _delattr__(self, name)可用來修改和刪除屬性。
它們的應用面更廣,可用於任意屬性。