欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Python類方法__init__和__del__構造、析構過程分析

 更新時間:2015年03月06日 10:10:41   投稿:junjie  
這篇文章主要介紹了Python類方法__init__和__del__構造、析構過程分析,本文分析了什么時候構造、什么時候析構、成員變量如何處理、Python中的共享成員函數(shù)如何訪問等問題,需要的朋友可以參考下

最近學習《Python參考手冊》學到Class部分,遇到了類的構造析構部分的問題:

1、什么時候構造?
2、什么時候析構?
3、成員變量如何處理?
4、Python中的共享成員函數(shù)如何訪問?
------------------------
探索過程:
1、經(jīng)過查找,Python中沒有專用的構造和析構函數(shù),但是一般可以在__init__和__del__分別完成初始化和刪除操作,可用這個替代構造和析構。還有一個__new__用來定制類的創(chuàng)建過程,不過需要一定的配置,此處不做討論。
2、類的成員函數(shù)默認都相當于是public的,但是默認開頭為__的為私有變量,雖然是私有,但是我們還可以通過一定的手段訪問到,即Python不存在真正的私有變量。如:

復制代碼 代碼如下:

__priValue = 0 # 會自動變形為"_類名__priValue"的成員變量

3、由于Python的特殊性,全局成員變量是共享的,所以類的實例不會為它專門分配內(nèi)容空間,類似于static,具體使用參看下面的例子。

測試1:

復制代碼 代碼如下:

# encoding:utf8

class NewClass(object):
    num_count = 0 # 所有的實例都共享此變量,即不單獨為每個實例分配
    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)試運行:

復制代碼 代碼如下:

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 是全局的,當每創(chuàng)建一個實例,__init__()被調(diào)用,num_count 的值增一,當程序結束后,所有的實例會被析構,即調(diào)用__del__() 但是此時引發(fā)了異常。查看異常為 “NoneType” 即 析構時NewClass 已經(jīng)被垃圾回收,所以會產(chǎn)生這樣的異常。

但是,疑問來了?為什么會這樣?按照C/C++等語言的經(jīng)驗,不應該這樣啊!經(jīng)過查找資料,發(fā)現(xiàn):

Python的垃圾回收過程與常用語言的不一樣,Python按照字典順序進行垃圾回收,而不是按照創(chuàng)建順序進行。所以當系統(tǒng)進行回收資源時,會按照類名A-Za-z的順序,依次進行,我們無法掌控這里的流程。

明白這些,我們做如下嘗試:

復制代碼 代碼如下:

# encoding:utf8

class NewClass(object):
    num_count = 0 # 所有的實例都共享此變量,即不單獨為每個實例分配
    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,一切按照我們預料的順序發(fā)生。
但是,我們總不能每次都手動回收吧?這么做Python自己的垃圾回收還有什么意義?

SO,繼續(xù)查找,我們還可以通過self.__class__訪問到類本身,然后再訪問自身的共享成員變量,即 self.__class__.num_count , 將類中的NewClass.num_count替換為self.__class__.num_count 編譯運行,如下:

復制代碼 代碼如下:

# encoding:utf8

class NewClass(object):
    num_count = 0 # 所有的實例都共享此變量,即不單獨為每個實例分配
    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"

結果:

復制代碼 代碼如下:

Hello 1
World 2
aaaa 3
Over
Del Hello 2
Del World 1
Del aaaa 0

Perfect!我們完美地處理了這個問題!

PS:

書上又提到了一些問題,在這里作補充(僅作為參考):

__new__()是唯一在實例創(chuàng)建之前執(zhí)行的方法,一般用在定義元類時使用。

del xxx 不會主動調(diào)用__del__方法,只有引用計數(shù)==0時,__del__()才會被執(zhí)行,并且定義了__del_()的實例無法被Python的循環(huán)垃圾收集器收集,所以盡量不要自定義__del__()。一般情況下,__del__() 不會破壞垃圾處理器。

實驗中發(fā)現(xiàn)垃圾回收自動調(diào)用了__del__, 這與書上所說又不符,不知是什么原因,需要繼續(xù)學習。

相關文章

  • python35種繪圖函數(shù)詳細總結

    python35種繪圖函數(shù)詳細總結

    Python有許多用于繪圖的函數(shù)和庫,比如Matplotlib,Plotly,Bokeh,Seaborn等,這只是一些常用的繪圖函數(shù)和庫,Python還有其他繪圖工具,如Pandas、ggplot等,選擇適合你需求的庫,可以根據(jù)你的數(shù)據(jù)類型、圖形需求和個人偏好來決定,本文給大家總結了python35種繪圖函數(shù)
    2023-08-08
  • Python實現(xiàn)人臉識別

    Python實現(xiàn)人臉識別

    這篇文章主要介紹了Python實現(xiàn)人臉識別,首選抓取多張圖片,從中獲取特征數(shù)據(jù)集和平均特征值然后寫入?csv?文件?-?計算特征數(shù)據(jù)集的歐式距離作對比,下面一起來看具體得實現(xiàn)過程吧
    2022-01-01
  • 在python中利用try..except來代替if..else的用法

    在python中利用try..except來代替if..else的用法

    今天小編就為大家分享一篇在python中利用try..except來代替if..else的用法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2019-12-12
  • 全面了解python中的類,對象,方法,屬性

    全面了解python中的類,對象,方法,屬性

    下面小編就為大家?guī)硪黄媪私鈖ython中的類,對象,方法,屬性。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2016-09-09
  • Django 允許局域網(wǎng)中的機器訪問你的主機操作

    Django 允許局域網(wǎng)中的機器訪問你的主機操作

    這篇文章主要介紹了Django 允許局域網(wǎng)中的機器訪問你的主機實例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-05-05
  • Python實現(xiàn)對PPT文件進行截圖操作的方法

    Python實現(xiàn)對PPT文件進行截圖操作的方法

    這篇文章主要介紹了Python實現(xiàn)對PPT文件進行截圖操作的方法,涉及Python操作幻燈片的相關技巧,非常具有實用價值,需要的朋友可以參考下
    2015-04-04
  • Python求一批字符串的最長公共前綴算法示例

    Python求一批字符串的最長公共前綴算法示例

    這篇文章主要介紹了Python求一批字符串的最長公共前綴算法,涉及Python針對字符串的遍歷、判斷、計算等相關操作技巧,需要的朋友可以參考下
    2019-03-03
  • python 通過字符串調(diào)用對象屬性或方法的實例講解

    python 通過字符串調(diào)用對象屬性或方法的實例講解

    下面小編就為大家分享一篇python 通過字符串調(diào)用對象屬性或方法的實例講解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-04-04
  • Python 完美解決 Import “模塊“ could not be resolved ...的問題

    Python 完美解決 Import “模塊“ could not&n

    這篇文章主要介紹了Python 完美解決 Import “模塊“ could not be resolved ...,本文給大家分享問題原因及解決方法,需要的朋友可以參考下
    2022-11-11
  • python中PyQuery庫用法分享

    python中PyQuery庫用法分享

    在本篇文章里小編給大家整理了一篇關于python中PyQuery庫用法內(nèi)容,有興趣的朋友們參考下。
    2021-01-01

最新評論