python 类属性和对象属性--定义和作用域
類屬性和對(duì)象屬性定義
Python一切皆對(duì)象(object),每個(gè)對(duì)象都可能有多個(gè)屬性(attribute)。Python的屬性有一套統(tǒng)一的管理方案。
對(duì)象的屬性可能來自于其類定義,叫做類屬性(class attribute)。類屬性可能來自類定義自身,也可能根據(jù)類定義繼承來的。
一個(gè)對(duì)象的屬性還可能是該對(duì)象實(shí)例定義的,叫做對(duì)象屬性(object attribute)。
對(duì)象的屬性儲(chǔ)存在對(duì)象的__dict__屬性中。
dir() 返回的僅是對(duì)象的屬性的一個(gè)名字類表,而__dict__返回的是一個(gè)字典,它的鍵(key)是屬性名,鍵值(value)是相應(yīng)的屬性對(duì)象的數(shù)據(jù)值。
chicken類繼承自bird類,而summer為chicken類的一個(gè)對(duì)象。
class bird(object):feather = Trueclass chicken(bird):fly = Falsedef __init__(self, age):self.age = age測(cè)試
summer = chicken(2)>>> print(bird.__dict__) 類屬性 {'__dict__': <attribute '__dict__' of 'bird' objects>, '__module__': '__main__', '__weakref__': <attribute '__weakref__' of 'bird' objects>, 'feather': True, '__doc__': None}>>> print(chicken.__dict__) 類屬性 {'fly': False, '__module__': '__main__', '__doc__': None, '__init__': <function __init__ at 0x2b91db476d70>}>>> print(summer.__dict__) 對(duì)象屬性 {'age': 2}當(dāng)我們有一個(gè)summer對(duì)象的時(shí)候,分別查詢summer對(duì)象、chicken類、bird類以及object類的屬性,就可以知道summer對(duì)象所有的__dict__,就可以找到通過對(duì)象summer可以調(diào)用和修改的所有屬性了。下面兩種屬性修改方法等效:
summer.__dict__['age'] = 3 summer.age = 5即時(shí)生成屬性
同一個(gè)對(duì)象的不同屬性之間可能存在依賴關(guān)系。當(dāng)某個(gè)屬性被修改時(shí),我們希望依賴于該屬性的其他屬性也同時(shí)變化。這時(shí),我們不能通過__dict__的方式來靜態(tài)的儲(chǔ)存屬性。Python提供了多種即時(shí)生成屬性的方法。其中一種稱為特性(property)。特性是特殊的屬性。比如我們?yōu)閏hicken類增加一個(gè)特性adult。當(dāng)對(duì)象的age超過1時(shí),adult為True;否則為False:
方案1 --> 稍顯麻煩
''' 遇到問題沒人解答?小編創(chuàng)建了一個(gè)Python學(xué)習(xí)交流QQ群:778463939 尋找有志同道合的小伙伴,互幫互助,群里還有不錯(cuò)的視頻學(xué)習(xí)教程和PDF電子書! ''' class chicken(bird):fly = Falsedef __init__(self, age):self.age = agedef getAdult(self):if self.age > 1.0: self.adult = Trueelse: self.adult = False獲得adult對(duì)象屬性的方法
>>> summer = chicken(2) #實(shí)例化 >>> summer.getAdult() #調(diào)用getAdult()方法,給adult屬性賦值 >>> summer.adult #獲取adult屬性方案2 --> 使用三元選擇,實(shí)用性較低
class bird(object):feather = True class chicken(bird):fly = Falsedef __init__(self, age):self.age = ageself.adult = True if self.age > 1.0 else False獲得adult對(duì)象屬性的方法
>>> summer = chicken(2) #實(shí)例化 >>> summer.adult #獲取adult屬性方案3 使用property,實(shí)用性更強(qiáng)
''' 遇到問題沒人解答?小編創(chuàng)建了一個(gè)Python學(xué)習(xí)交流QQ群:778463939 尋找有志同道合的小伙伴,互幫互助,群里還有不錯(cuò)的視頻學(xué)習(xí)教程和PDF電子書! ''' class bird(object):feather = Trueclass chicken(bird):fly = Falsedef __init__(self, age):self.age = agedef getAdult(self):if self.age > 1.0: return Trueelse: return Falseadult = property(getAdult)獲得adult對(duì)象屬性的方法
>>> summer = chicken(2) #實(shí)例化 >>> summer.adult #獲取adult屬性????
class num(object):def __init__(self, value):self.value = valuedef getNeg(self):return -self.valuedef setNeg(self, value):self.value = -valuedef delNeg(self):print("value also deleted")del self.valueneg = 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上面的num為一個(gè)數(shù)字,而neg為一個(gè)特性,用來表示數(shù)字的負(fù)數(shù)。當(dāng)一個(gè)數(shù)字確定的時(shí)候,它的負(fù)數(shù)總是確定的;而當(dāng)我們修改一個(gè)數(shù)的負(fù)數(shù)時(shí),它本身的值也應(yīng)該變化。這兩點(diǎn)由getNeg和setNeg來實(shí)現(xiàn)。而delNeg表示的是,如果刪除特性neg,那么應(yīng)該執(zhí)行的操作是刪除屬性value。property()的最后一個(gè)參數(shù)(“I’m negative”)為特性negative的說明文檔。
方案4 使用特殊方法__getattr__
''' 遇到問題沒人解答?小編創(chuàng)建了一個(gè)Python學(xué)習(xí)交流QQ群:778463939 尋找有志同道合的小伙伴,互幫互助,群里還有不錯(cuò)的視頻學(xué)習(xí)教程和PDF電子書! ''' class bird(object):feather = True class chicken(bird):fly = Falsedef __init__(self, age):self.age = agedef __getattr__(self, name):if name == 'adult':if self.age > 1.0: return Trueelse: return Falseelse: raise AttributeError(name)每個(gè)特性需要有自己的處理函數(shù),而__getattr__可以將所有的即時(shí)生成屬性放在同一個(gè)函數(shù)中處理。__getattr__可以根據(jù)函數(shù)名區(qū)別處理不同的屬性。比如上面我們查詢屬性名male的時(shí)候,raise AttributeError。
(Python中還有一個(gè)__getattribute__特殊方法,用于查詢?nèi)我鈱傩浴_getattr__只能用來查詢不在__dict__系統(tǒng)中的屬性)
__setattr__(self, name, value)和__delattr__(self, name)可用于修改和刪除屬性。它們的應(yīng)用面更廣,可用于任意屬性。
即時(shí)生成屬性的其他方式
即時(shí)生成屬性還可以使用其他的方式,比如descriptor(descriptor類實(shí)際上是property()函數(shù)的底層,property()實(shí)際上創(chuàng)建了一個(gè)該類的對(duì)象)。有興趣可以進(jìn)一步查閱。
類屬性和對(duì)象屬性的作用域
類屬性:
- 公有屬性,私有屬性,內(nèi)置屬性
對(duì)象屬性:
- 公有屬性,私有屬性,內(nèi)置屬性,方法變量,全局變量
根據(jù)屬性不同,起作用范圍也不同。如下為圖示和說明:
class MyClass(object):"""類公有屬性:在類中使用self.var1調(diào)用,類外使MyClass.var1調(diào)用,也可以通過實(shí)例調(diào)用 instance.var1"""var1 = 'class public attr: var1'"""類私有屬性:在類中使用self.__var2調(diào)用, 類外使用MyClass._MyClass__var2調(diào)用,也可以通過實(shí)例調(diào)用instance._MyClass__var2"""__var2 = 'class private attr: __var2'def method(self):"""在方法method內(nèi)有效"""var3 = 'method local attr: var3'print(var3)"""對(duì)象公有屬性:方法調(diào)用后生效,在類內(nèi)方法間通過self.var4調(diào)用,在類外通過instance.var4"""self.var4 = 'object public attr: self.var4'print(self.var4)"""對(duì)象私有屬性:方法調(diào)用后生效, 在方法間內(nèi)通過self.__var5調(diào)用,類內(nèi)其他方法不可調(diào)用,在類外通過instance.__var5"""self.__var5 = 'object private attr: self.__var5'print(self.__var5)def method2(self):print(self.var1)print(self.__var2)print(self.var4)"""通過類調(diào)用類屬性""" >>> MyClass.var1 'class public attr: var1' >>> MyClass._MyClass__var2 'class private attr: __var2' """通過對(duì)象調(diào)用類屬性""" >>> obj = MyClass() >>> obj.var1 'class public attr: var1' >>> obj._MyClass__var2 'class private attr: __var2' >>> obj.var4 'object public attr: self.var4' >>> obj._MyClass__var5 'object private attr: self.__var5'總結(jié)
__dict__分層存儲(chǔ)屬性。每一層的__dict__只存儲(chǔ)該層新增的屬性。子類不需要重復(fù)存儲(chǔ)父類中的屬性。
即時(shí)生成屬性是值得了解的概念。在Python開發(fā)中,你有可能使用這種方法來更合理的管理對(duì)象的屬性。
總結(jié)
以上是生活随笔為你收集整理的python 类属性和对象属性--定义和作用域的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: python基础教程: 自定义函数
- 下一篇: Python枚举类型的使用