Python中的類(lèi)屬性與實(shí)例屬性區(qū)別詳解
類(lèi)屬性與實(shí)例屬性
首先我們簡(jiǎn)要說(shuō)下類(lèi)屬性與實(shí)例屬性在概念上的不同之處:
類(lèi)屬性是在類(lèi)中定義的屬性,它是和這個(gè)類(lèi)所綁定的,這個(gè)類(lèi)中的所有對(duì)象都可以訪問(wèn)。
訪問(wèn)時(shí)可以通過(guò)類(lèi)名來(lái)訪問(wèn),也可以通過(guò)實(shí)例名來(lái)訪問(wèn)。
實(shí)例屬性是與類(lèi)的實(shí)例相關(guān)聯(lián)的數(shù)據(jù)值,是這個(gè)實(shí)例私有的,只有這個(gè)對(duì)象自己可以訪問(wèn)。
當(dāng)一個(gè)實(shí)例被釋放后,它的屬性同時(shí)也被清除了。
然后我們通過(guò)一個(gè)具體的例子,來(lái)看下在訪問(wèn)類(lèi)屬性和實(shí)例屬性時(shí),Python是怎么進(jìn)行操作的。
# 定義了類(lèi)之后,Python就會(huì)為類(lèi)分配一塊內(nèi)存空間,里面放它的相關(guān)屬性和方法。
# 這里在類(lèi)中定一個(gè)了一個(gè)類(lèi)屬性,相當(dāng)于在Person類(lèi)的內(nèi)存空間中有了值為10的age屬性。
class Person():
age = 10
# 給類(lèi)名加上函數(shù)調(diào)用符號(hào),就相當(dāng)于創(chuàng)建了一個(gè)對(duì)象。
# 現(xiàn)在我們分別創(chuàng)建了一個(gè)叫沈騰的人和一個(gè)叫馬麗的人。
# 創(chuàng)建后,Python也會(huì)為對(duì)象分別分配內(nèi)存空間
st = Person()
ml = Person()
# 使用對(duì)象訪問(wèn)類(lèi)屬性,它會(huì)先去對(duì)象的內(nèi)存空間里面找age屬性,如果沒(méi)有,就向上找對(duì)象所屬類(lèi)的內(nèi)存空間。
# 而在這里它并不是直接取了對(duì)象的內(nèi)存空間的age屬性,因?yàn)榇藭r(shí)對(duì)象的內(nèi)存空間里面還沒(méi)有age屬性。
# 這里實(shí)際上是對(duì)象訪問(wèn)了類(lèi)的內(nèi)存空間,從類(lèi)內(nèi)存空間中取出來(lái)age屬性的值,并打印出來(lái)
# 所以這里沈騰對(duì)象屬性值和馬麗對(duì)象屬性值都是類(lèi)屬性值10
print("st對(duì)象:%s" % st.age)
print("ml對(duì)象:%s" % ml.age)
# 要記住一點(diǎn),只要給對(duì)象有了賦值操作,那么就相當(dāng)于給對(duì)象的內(nèi)存空間中動(dòng)態(tài)創(chuàng)建了一個(gè)屬性,所以這里此時(shí)沈騰對(duì)象的內(nèi)存空間中有了一個(gè)age屬性了,是屬于這個(gè)對(duì)象的屬性,也就是我們所說(shuō)的實(shí)例屬性。
# 那么此時(shí)st.age會(huì)先在對(duì)象內(nèi)存空間中找age屬性,找到了,就不會(huì)再去類(lèi)內(nèi)存空間中找,所以此時(shí)st.age訪問(wèn)的是對(duì)象內(nèi)存空間的age屬性。
# 所以此時(shí)沈騰對(duì)象屬性值是沈騰對(duì)象內(nèi)存空間age屬性的值,也就是12。而非類(lèi)屬性值10。
st.age = 12
print("st對(duì)象:%s" % st.age)
# 因?yàn)轳R麗對(duì)象還沒(méi)有進(jìn)行賦值操作,所以它還沒(méi)有在它自己的內(nèi)存空間中動(dòng)態(tài)創(chuàng)建age屬性。所以它這里訪問(wèn)的依舊是類(lèi)內(nèi)存空間中的age屬性值。
# 而上述語(yǔ)句改變的是沈騰內(nèi)存空間中age的值,所以不會(huì)影響到馬麗對(duì)象屬性值。
# 這里打印馬麗對(duì)象屬性值也就是類(lèi)的屬性值10。
print("ml對(duì)象:%s" % ml.age)
# 因?yàn)樯厦娴馁x值語(yǔ)句改變的是對(duì)象的內(nèi)存空間,所以類(lèi)屬性的值其實(shí)并沒(méi)有改變,依舊是原來(lái)的值10。
print("類(lèi)屬性值:%s" % Person.age) 我們看下具體的打印結(jié)果和我們分析的是否一致
可以看到,在沒(méi)有st.age沒(méi)有執(zhí)行之前的打印值都是類(lèi)屬性的值10,而st.age=12執(zhí)行后,st.age值就變成了12,而ml.age和Person.age都沒(méi)有變,依舊是類(lèi)屬性的10。

我們?cè)偻ㄟ^(guò)圖解的形式理解下:


還有一點(diǎn)需要說(shuō)明,我們可以在類(lèi)的構(gòu)造方法__init__中對(duì)對(duì)象的屬性進(jìn)行初始化,這里也是相當(dāng)于對(duì)對(duì)象的屬性進(jìn)行了賦值操作,所以也是在對(duì)象的內(nèi)存空間中動(dòng)態(tài)的創(chuàng)建了實(shí)例屬性。
因?yàn)閟elf參數(shù)就相當(dāng)于對(duì)象自己,self.age在對(duì)象創(chuàng)建后就相當(dāng)于對(duì)象.age,如st.age,ml.age,和上面例子原理是一樣的。
class Person():
age = 10
def __init__(self, age):
self.age = age
# 因?yàn)樵跇?gòu)造方法中就已經(jīng)給self.age屬性賦值了。
# 所以創(chuàng)建對(duì)象時(shí),沈騰對(duì)象內(nèi)存空間中和馬麗對(duì)象內(nèi)存空間中就都有了age屬性,這是屬于對(duì)象的屬性。
# 所以以后在使用st.age和ml.age時(shí),訪問(wèn)到的都是對(duì)象內(nèi)存空間中的age值了。
st = Person(18)
ml = Person(9)
# 這里就訪問(wèn)的是對(duì)象內(nèi)存空間中的age屬性值了
# 所以分別打印18和9
print("st對(duì)象:%s" % st.age)
print("ml對(duì)象:%s" % ml.age)
# 這里給沈騰對(duì)象的age屬性重新賦值,改變的也僅是沈騰對(duì)象內(nèi)存空間中的age屬性值
# 不會(huì)改變馬麗對(duì)象內(nèi)存空間age屬性值和類(lèi)內(nèi)存空間age屬性值。
st.age = 12
print("st對(duì)象:%s" % st.age)
print("ml對(duì)象:%s" % ml.age)
print("類(lèi)屬性值:%s" % Person.age) 我們來(lái)看下打印結(jié)果:

這個(gè)也來(lái)個(gè)圖解吧

總結(jié)下:
- 類(lèi)屬性在類(lèi)創(chuàng)建時(shí)就存在于類(lèi)的內(nèi)存空間中。
- 如果類(lèi)的構(gòu)造函數(shù)中沒(méi)有初始化對(duì)象屬性,那么對(duì)象在創(chuàng)建時(shí)內(nèi)存空間是沒(méi)有這個(gè)屬性的。
- 實(shí)例屬性是通過(guò)賦值語(yǔ)句來(lái)動(dòng)態(tài)創(chuàng)建的。如果沒(méi)有動(dòng)態(tài)創(chuàng)建,通過(guò)對(duì)象訪問(wèn)和類(lèi)同名的屬性時(shí),會(huì)現(xiàn)在對(duì)象內(nèi)存空間中查找是否有該屬性,沒(méi)有就去類(lèi)內(nèi)存空間中查找。
- 如果已經(jīng)動(dòng)態(tài)創(chuàng)建了實(shí)例屬性,那么Python使用對(duì)象訪問(wèn)和類(lèi)同名的屬性時(shí),是一定先訪問(wèn)對(duì)象內(nèi)存空間中的實(shí)例屬性的。
到此這篇關(guān)于Python中的類(lèi)屬性與實(shí)例屬性區(qū)別詳解的文章就介紹到這了,更多相關(guān)Python類(lèi)屬性與實(shí)例屬性內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Python+Selenium自動(dòng)化實(shí)現(xiàn)分頁(yè)(pagination)處理
這篇文章主要為大家詳細(xì)介紹了Python+Selenium自動(dòng)化實(shí)現(xiàn)分頁(yè)pagination處理的方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-03-03
Pycharm搭建Django項(xiàng)目詳細(xì)教程(看完這一篇就夠了)
這篇文章主要給大家介紹了關(guān)于Pycharm搭建Django項(xiàng)目的詳細(xì)教程,想要學(xué)習(xí)的小伙伴看完這一篇就夠了,pycharm是一種Python?IDE,帶有一整套可以幫助用戶在使用Python語(yǔ)言開(kāi)發(fā)時(shí)提高其效率的工具,需要的朋友可以參考下2023-11-11
Python實(shí)現(xiàn)命令行通訊錄實(shí)例教程
這篇文章主要介紹怎樣編寫(xiě)了一段命令行通訊錄的小程序。下面是編寫(xiě)的思路以及代碼,歡迎感興趣的同學(xué)交流探討。2016-08-08
python實(shí)現(xiàn)簡(jiǎn)單淘寶秒殺功能
這篇文章主要為大家詳細(xì)介紹了python實(shí)現(xiàn)簡(jiǎn)單淘寶秒殺功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-05-05
Python數(shù)據(jù)存儲(chǔ)之 h5py詳解
今天小編就為大家分享一篇Python數(shù)據(jù)存儲(chǔ)之 h5py詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-12-12
python3實(shí)現(xiàn)elasticsearch批量更新數(shù)據(jù)
今天小編就為大家分享一篇python3實(shí)現(xiàn)elasticsearch批量更新數(shù)據(jù),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-12-12
pycharm自動(dòng)生成文件注釋和函數(shù)注釋
這篇文章主要介紹了pycharm自動(dòng)生成文件注釋和函數(shù)注釋的方法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-07-07
python按鍵按住不放持續(xù)響應(yīng)的實(shí)例代碼
今天小編就為大家分享一篇python按鍵按住不放持續(xù)響應(yīng)的實(shí)例代碼,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-07-07

