詳解python 內(nèi)存優(yōu)化
寫(xiě)在之前
圍繞類(lèi)的話(huà)題,說(shuō)是說(shuō)不完的,僅在特殊方法,除了我們?cè)谇懊嬗龅竭^(guò)的 __init__(),__new__(),__str__() 等之外還有很多。雖然它們只是在某些特殊的場(chǎng)景中才會(huì)用到,但是學(xué)會(huì)它們卻可以成為你熟悉這門(mén)語(yǔ)言路上的鋪路石。
所以我會(huì)在試圖介紹一些「黑魔法」,讓大家多多感受一下 Python 的魅力所在,俗話(huà)說(shuō)「藝多不壓身」就是這個(gè)道理了。
內(nèi)存優(yōu)化
首先先讓我們從復(fù)習(xí)前面的類(lèi)屬性和實(shí)例屬性的知識(shí)來(lái)引出另一個(gè)特殊方法:
>>> class Sample: ... name = 'rocky' ...
就像前面的文章我們所說(shuō)的,每個(gè)類(lèi)都有一個(gè) __dict__() 屬性,它包含了當(dāng)前類(lèi)的類(lèi)屬性:
>>> Sample.__dict__ mappingproxy({'__module__': '__main__', 'name': 'rocky', '__dict__': <attribute '__dict__' of 'Sample' objects>, '__weakref__': <attribute '__weakref__' of 'Sample' objects>, '__doc__': None}) >>> Sample.name 'rocky'
同樣,如果我們創(chuàng)建了實(shí)例,每個(gè)實(shí)例也有一個(gè) __dict__ 屬性,它里面就是當(dāng)前的實(shí)例屬性:
>>> a = Sample() >>> a.__dict__ {} >>> a.age = 23 >>> a.__dict__ {'age': 23}
上面的操作可以看出,當(dāng)實(shí)例剛剛創(chuàng)建的時(shí)候,__dict__ 是空的,只有創(chuàng)建了實(shí)例屬性以后,它才包含其內(nèi)容。實(shí)例的 __dict__ 和類(lèi)的 __dict__ 是有所區(qū)別的,即實(shí)例屬性和類(lèi)屬性是不同的。
從理論上來(lái)說(shuō),我們可以根據(jù)一個(gè)類(lèi)創(chuàng)建無(wú)數(shù)的實(shí)例,新建一個(gè)實(shí)例以后,又創(chuàng)建了一個(gè)新的 __dict__,這將是一個(gè)很可怕的事情,雖然每個(gè) __dict__ 所占的內(nèi)存空間很小,當(dāng)然這件事事實(shí)上是不會(huì)出現(xiàn)的。但是程序不能建立在這種不可靠的猜測(cè)的基礎(chǔ)上,程序要對(duì)過(guò)程有明確的控制。
所以就要有一種方法能夠控制 __dict__,于是「__slots__」應(yīng)運(yùn)而生。
>>> class Nature: ... __slots__ = ('tree','flower') ... >>> dir(Nature) ['__class__', '__delattr__', '__dir__', '__doc__', '__eq__', '__format__', '__ge__', '__getattribute__', '__gt__', '__hash__', '__init__', '__init_subclass__', '__le__', '__lt__', '__module__', '__ne__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__slots__', '__str__', '__subclasshook__', 'flower', 'tree']
我們仔細(xì)來(lái)看 dir() 的結(jié)果,發(fā)現(xiàn) __dict__ 屬性沒(méi)有了,也就是說(shuō) __slots__ 把 __dict__ 擠出去了,它進(jìn)入了類(lèi)的屬性。
>>> Nature.__slots__ ('tree', 'flower')
從這里可以看出,類(lèi) Nature 有且僅有兩個(gè)屬性。從類(lèi)的角度來(lái)看,其類(lèi)屬性只有這兩個(gè);從實(shí)例的角度來(lái)看,其實(shí)例屬性也只有這兩個(gè)。
>>> Nature.tree = 'liushu' >>> Nature.tree 'liushu' >>> Nature.tree = 'lishu' >>> Nature.tree 'lishu'
通過(guò)類(lèi)可以對(duì)屬性進(jìn)行賦值和修改,這個(gè)似乎和以前的類(lèi)屬性沒(méi)有什么區(qū)別,別著急,繼續(xù)往下看就看到區(qū)別了:
>>> x = Nature() >>> x.__slots__ ('tree', 'flower') >>> y = Nature() >>> y.__slots__ ('tree', 'flower') >>> id(x.__slots__) 4531629384 >>> id(y.__slots__) 4531629384
你看,實(shí)例化以后,實(shí)例的 __slots__ 和類(lèi)的 __slots__ 完全一樣,這跟前面的 __dict__ 大不一樣了。并且我們建立了兩個(gè)實(shí)例,結(jié)果發(fā)現(xiàn)兩個(gè)實(shí)例的 __slots__ 在內(nèi)存中居然是一個(gè),或者可以說(shuō)是增加實(shí)例時(shí) __slots__ 并不增加。
>>> x.tree 'lishu' >>> y.tree 'lishu'
既然類(lèi)屬性已經(jīng)賦值,那么通過(guò)任何一個(gè)實(shí)例屬性都能得到同樣的值,不過(guò)這時(shí)候不能通過(guò)實(shí)例修改此屬性的值。
>>> x.tree = 'taoshu' Traceback (most recent call last): File "<stdin>", line 1, in <module> AttributeError: 'Nature' object attribute 'tree' is read-only
對(duì)實(shí)例屬性來(lái)說(shuō),類(lèi)的靜態(tài)數(shù)據(jù)是只讀的,不能修改,只有通過(guò)類(lèi)屬性才能修改。但對(duì)于尚未賦值的屬性,能夠通過(guò)實(shí)例賦值。
>>> x.flower = 'rose' >>> x.flower 'rose' >>> x.flower = 'moli'
顯然通過(guò)實(shí)例操作的屬性,也能夠通過(guò)實(shí)例修改,但是實(shí)例屬性的值并不能夠修改類(lèi)屬性的值
Nature.flower <member 'flower' of 'Nature' objects>
由上面可以看出,實(shí)例屬性的值并沒(méi)有傳回給類(lèi)屬性,也可以理解為新建了一個(gè)同名字的實(shí)例屬性,如果再給類(lèi)屬性賦值的話(huà),則會(huì)像下面一樣:
>>> Nature.flower = 'huaihua' >>> x.flower 'huaihua'
類(lèi)屬性對(duì)實(shí)例屬性具有決定作用,對(duì)實(shí)例而言,通過(guò)類(lèi)所定義的屬性都是只讀的。
__slots__ 已經(jīng)把實(shí)例屬性牢牢的看管起來(lái),只能是指定的屬性,如果想要增加屬性的話(huà),只能通過(guò)類(lèi)屬性來(lái)實(shí)現(xiàn),所以 __slots__ 的一個(gè)重要作用就是優(yōu)化了內(nèi)存。
寫(xiě)在之后
當(dāng)然了,__slots__ 還能加快屬性加載速度,這個(gè)不是本文的重點(diǎn),所以不做過(guò)多的介紹,感興趣的可以去 Google 一下。
今天的文章就到這里啦,明天講一下「屬性攔截」,又是新的一周,燥起來(lái)!
如果你覺(jué)得文章對(duì)你有幫助的話(huà),歡迎點(diǎn)贊轉(zhuǎn)發(fā),讓更多的人看到,謝謝啦。
The end。
以上就是詳解python 內(nèi)存優(yōu)化的詳細(xì)內(nèi)容,更多關(guān)于python 內(nèi)存優(yōu)化的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
pygame用blit()實(shí)現(xiàn)動(dòng)畫(huà)效果的示例代碼
這篇文章主要介紹了pygame用blit()實(shí)現(xiàn)動(dòng)畫(huà)效果的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-05-05使用 prometheus python 庫(kù)編寫(xiě)自定義指標(biāo)的方法(完整代碼)
這篇文章主要介紹了使用 prometheus python 庫(kù)編寫(xiě)自定義指標(biāo)的方法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-06-06對(duì)django中render()與render_to_response()的區(qū)別詳解
今天小編就為大家分享一篇對(duì)django中render()與render_to_response()的區(qū)別詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-10-10