概述
GC作為現(xiàn)代編程語(yǔ)言的自動(dòng)內(nèi)存管理機(jī)制,專(zhuān)注于兩件事:1. 找到內(nèi)存中無(wú)用的垃圾資源 2. 清除這些垃圾并把內(nèi)存讓出來(lái)給其他對(duì)象使用乃正。 在Python中,它在每個(gè)對(duì)象中保持了一個(gè)計(jì)數(shù)器,用于記錄指向該對(duì)象的的引用的個(gè)數(shù)。一旦這個(gè)計(jì)數(shù)器為0時(shí)护锤,則立即回收該對(duì)象,對(duì)象占用的內(nèi)存空間將被釋放减拭。
引用計(jì)數(shù)
我們可以利用簡(jiǎn)單的變量引用和銷(xiāo)毀窺見(jiàn)引用計(jì)數(shù)過(guò)程蔽豺。
增加引用計(jì)數(shù)
增加引用計(jì)數(shù)的方式多種区丑,即對(duì)象進(jìn)行引用拧粪,那么計(jì)數(shù)器都會(huì)+1
# 創(chuàng)建第一個(gè)引用
a = 3
# 用其他變量名引用
b = a
# 成為一個(gè)容器的對(duì)象
L = [1, a]
# 作為參數(shù)傳遞
str(a)
減少引用計(jì)數(shù)
同理,以下是減少引用計(jì)數(shù)的一些方法
# 一個(gè)本地引用離開(kāi)了其作用范圍沧侥。比如`str()`函數(shù)結(jié)束時(shí)
str(a)
# 對(duì)象的別名被顯式銷(xiāo)毀
del a
# 對(duì)象的一個(gè)別名被復(fù)制給其他對(duì)象
a = 'Python'
# 對(duì)象從一個(gè)窗口對(duì)象中移除
L.remove(a)
# 窗口對(duì)象本身被銷(xiāo)毀
del L
循環(huán)引用問(wèn)題
什么是循環(huán)引用可霎?A和B相互引用而再?zèng)]有外部引用A與B中的任何一個(gè),它們的引用計(jì)數(shù)雖然都為1宴杀,但顯然應(yīng)該被回收癣朗。
# 對(duì)象a的引用計(jì)數(shù)為 1
a = {}
# 對(duì)象B的引用計(jì)數(shù)為 1
b = {}
# B的引用計(jì)數(shù)增1
a['b'] = b
# A的引用計(jì)數(shù)增1
b['a'] = a
# A的引用減 1,最后A對(duì)象的引用為 1
del a
# B的引用減 1, 最后B對(duì)象的引用為 1
del b
在這個(gè)例子中程序執(zhí)行完del語(yǔ)句后旺罢,A旷余、B對(duì)象已經(jīng)沒(méi)有任何引用指向這兩個(gè)對(duì)象,但這兩個(gè)對(duì)象卻還各自引用這對(duì)象扁达,雖然兩個(gè)對(duì)象已經(jīng)被del
了正卧,即我們不能再使用這兩個(gè)對(duì)象,即垃圾對(duì)象跪解,但是他們的引用計(jì)數(shù)并沒(méi)有減少到零炉旷。即根據(jù)引用計(jì)數(shù)機(jī)制,他們并不會(huì)被回收,且會(huì)一直駐留在內(nèi)存中窘行,造成內(nèi)存泄漏饥追。為了解決對(duì)象的循環(huán)引用問(wèn)題,而Python引入了標(biāo)記-清除和分代回收兩種GC機(jī)制來(lái)解決優(yōu)化此問(wèn)題罐盔。