Python 高階之 __attr__ 物件屬性

巴扎黑
發布: 2016-12-08 09:29:53
原創
1384 人瀏覽過

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__': , '__module__': '__we ': , 'feather': True, '__doc__': None}

{'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 &#39;<--init&#39;
    def getNeg(self):
print &#39;<--getNeg&#39;
return self.value * -1
    def setNeg(self, value):
print &#39;<--setNeg&#39;
self.value = (-1) * value
    def delNeg(self):
print("value also deleted")
del self.value
    neg = property(getNeg, setNeg, delNeg, "I&#39;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 == &#39;adult&#39;:
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)可用來修改和刪除屬性。

它們的應用面更廣,可用於任意屬性。


相關標籤:
來源:php.cn
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
熱門教學
更多>
最新下載
更多>
網站特效
網站源碼
網站素材
前端模板