Python類方法__init__和__del__構(gòu)造、析構(gòu)過程分析
最近學(xué)習(xí)《Python參考手冊(cè)》學(xué)到Class部分,遇到了類的構(gòu)造析構(gòu)部分的問題:
1、什么時(shí)候構(gòu)造?
2、什么時(shí)候析構(gòu)?
3、成員變量如何處理?
4、Python中的共享成員函數(shù)如何訪問?
------------------------
探索過程:
1、經(jīng)過查找,Python中沒有專用的構(gòu)造和析構(gòu)函數(shù),但是一般可以在__init__和__del__分別完成初始化和刪除操作,可用這個(gè)替代構(gòu)造和析構(gòu)。還有一個(gè)__new__用來定制類的創(chuàng)建過程,不過需要一定的配置,此處不做討論。
2、類的成員函數(shù)默認(rèn)都相當(dāng)于是public的,但是默認(rèn)開頭為__的為私有變量,雖然是私有,但是我們還可以通過一定的手段訪問到,即Python不存在真正的私有變量。如:
__priValue = 0 # 會(huì)自動(dòng)變形為"_類名__priValue"的成員變量
3、由于Python的特殊性,全局成員變量是共享的,所以類的實(shí)例不會(huì)為它專門分配內(nèi)容空間,類似于static,具體使用參看下面的例子。
測(cè)試1:
# encoding:utf8
class NewClass(object):
num_count = 0 # 所有的實(shí)例都共享此變量,即不單獨(dú)為每個(gè)實(shí)例分配
def __init__(self,name):
self.name = name
NewClass.num_count += 1
print name,NewClass.num_count
def __del__(self):
NewClass.num_count -= 1
print "Del",self.name,NewClass.num_count
def test():
print "aa"
aa = NewClass("Hello")
bb = NewClass("World")
cc = NewClass("aaaa")
print "Over"
調(diào)試運(yùn)行:
Hello 1
World 2
aaaa 3
Over
DeException l Hello 2
AttributeError: "'NoneType' object has no attribute 'num_count'" in <bound method NewClass.__del__ of <__main__.NewClass object at 0x01AF18D0>> ignored
Exception AttributeError: "'NoneType' object has no attribute 'num_count'" in <bound method NewClass.__del__ of <__main__.NewClass object at 0x01AF1970>> ignored
我們發(fā)現(xiàn),num_count 是全局的,當(dāng)每創(chuàng)建一個(gè)實(shí)例,__init__()被調(diào)用,num_count 的值增一,當(dāng)程序結(jié)束后,所有的實(shí)例會(huì)被析構(gòu),即調(diào)用__del__() 但是此時(shí)引發(fā)了異常。查看異常為 “NoneType” 即 析構(gòu)時(shí)NewClass 已經(jīng)被垃圾回收,所以會(huì)產(chǎn)生這樣的異常。
但是,疑問來了?為什么會(huì)這樣?按照C/C++等語言的經(jīng)驗(yàn),不應(yīng)該這樣?。〗?jīng)過查找資料,發(fā)現(xiàn):
Python的垃圾回收過程與常用語言的不一樣,Python按照字典順序進(jìn)行垃圾回收,而不是按照創(chuàng)建順序進(jìn)行。所以當(dāng)系統(tǒng)進(jìn)行回收資源時(shí),會(huì)按照類名A-Za-z的順序,依次進(jìn)行,我們無法掌控這里的流程。
明白這些,我們做如下嘗試:
# encoding:utf8
class NewClass(object):
num_count = 0 # 所有的實(shí)例都共享此變量,即不單獨(dú)為每個(gè)實(shí)例分配
def __init__(self,name):
self.name = name
NewClass.num_count += 1
print name,NewClass.num_count
def __del__(self):
NewClass.num_count -= 1
print "Del",self.name,NewClass.num_count
def test():
print "aa"
aa = NewClass("Hello")
bb = NewClass("World")
cc = NewClass("aaaa")
del aa
del bb
del cc
print "Over"
調(diào)試輸出:
Hello 1
World 2
aaaa 3
Del Hello 2
Del World 1
Del aaaa 0
Over
OK,一切按照我們預(yù)料的順序發(fā)生。
但是,我們總不能每次都手動(dòng)回收吧?這么做Python自己的垃圾回收還有什么意義?
SO,繼續(xù)查找,我們還可以通過self.__class__訪問到類本身,然后再訪問自身的共享成員變量,即 self.__class__.num_count , 將類中的NewClass.num_count替換為self.__class__.num_count 編譯運(yùn)行,如下:
# encoding:utf8
class NewClass(object):
num_count = 0 # 所有的實(shí)例都共享此變量,即不單獨(dú)為每個(gè)實(shí)例分配
def __init__(self,name):
self.name = name
self.__class__.num_count += 1
print name,NewClass.num_count
def __del__(self):
self.__class__.num_count -= 1
print "Del",self.name,self.__class__.num_count
def test():
print "aa"
aa = NewClass("Hello")
bb = NewClass("World")
cc = NewClass("aaaa")
print "Over"
結(jié)果:
Hello 1
World 2
aaaa 3
Over
Del Hello 2
Del World 1
Del aaaa 0
Perfect!我們完美地處理了這個(gè)問題!
PS:
書上又提到了一些問題,在這里作補(bǔ)充(僅作為參考):
__new__()是唯一在實(shí)例創(chuàng)建之前執(zhí)行的方法,一般用在定義元類時(shí)使用。
del xxx 不會(huì)主動(dòng)調(diào)用__del__方法,只有引用計(jì)數(shù)==0時(shí),__del__()才會(huì)被執(zhí)行,并且定義了__del_()的實(shí)例無法被Python的循環(huán)垃圾收集器收集,所以盡量不要自定義__del__()。一般情況下,__del__() 不會(huì)破壞垃圾處理器。
實(shí)驗(yàn)中發(fā)現(xiàn)垃圾回收自動(dòng)調(diào)用了__del__, 這與書上所說又不符,不知是什么原因,需要繼續(xù)學(xué)習(xí)。
- 詳細(xì)解讀Python中的__init__()方法
- python中的__init__ 、__new__、__call__小結(jié)
- Python中__init__.py文件的作用詳解
- 詳解Python中的__new__、__init__、__call__三個(gè)特殊方法
- python中子類繼承父類的__init__方法實(shí)例
- 淺談Python類里的__init__方法函數(shù),Python類的構(gòu)造函數(shù)
- Python中__new__與__init__方法的區(qū)別詳解
- python類繼承與子類實(shí)例初始化用法分析
- Python中類的初始化特殊方法
- python使用 __init__初始化操作簡(jiǎn)單示例
相關(guān)文章
Python答題卡識(shí)別并給出分?jǐn)?shù)的實(shí)現(xiàn)代碼
本文帶領(lǐng)大家學(xué)習(xí)Python答題卡識(shí)別并給出分?jǐn)?shù)的實(shí)現(xiàn)代碼,代碼實(shí)現(xiàn)思路清晰,簡(jiǎn)單易懂,Python識(shí)別答題卡相關(guān)知識(shí)感興趣的朋友一起看看吧2021-06-06
Python?imgaug庫安裝與使用教程(圖片加模糊光雨雪霧等特效)
imgaug機(jī)器學(xué)習(xí)實(shí)驗(yàn)中的圖像增強(qiáng)庫,特別是卷積神經(jīng)網(wǎng)絡(luò),支持以多種不同方式增強(qiáng)圖像、關(guān)鍵點(diǎn)/地標(biāo)、邊界框、熱圖和分割圖,這篇文章主要介紹了Python?imgaug庫?安裝與使用教程(圖片加模糊光雨雪霧等特效),需要的朋友可以參考下2022-11-11
人工智能學(xué)習(xí)pyTorch的ResNet殘差模塊示例詳解
這篇文章主要為大家介紹了人工智能學(xué)習(xí)pyTorch的ResNet殘差模塊示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步2021-11-11
tensorflow入門:tfrecord 和tf.data.TFRecordDataset的使用
今天小編就為大家分享一篇tensorflow入門:tfrecord 和tf.data.TFRecordDataset的使用,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-01-01
python實(shí)現(xiàn)對(duì)doc,txt,xls文檔的讀寫操作
這篇文章主要介紹了python實(shí)現(xiàn)對(duì)doc,txt,xls文檔的讀寫操作,正如標(biāo)題所見,文章包括三個(gè)部分python實(shí)現(xiàn)對(duì)doc文檔的讀取、python實(shí)現(xiàn)對(duì)txt文檔的讀取和python實(shí)現(xiàn)對(duì)xls表格的讀取,需要的朋友可以參考一下2022-04-04
Python中使用Queue和Condition進(jìn)行線程同步的方法
這篇文章主要介紹了Python中使用Queue模塊和Condition對(duì)象進(jìn)行線程同步的方法,配合threading模塊下的線程編程進(jìn)行操作的實(shí)例,需要的朋友可以參考下2016-01-01

