a = 1
a是引用,1是對象。Python緩存整數(shù)和短字符串,對象只有一份,但長字符串和其他對象(列表字典)則有很多對象(賦值語句創(chuàng)建新的對象)。
from sys import getrefcount a=[1,2,3] print(getfrecount(a))
返回4,當(dāng)使用某個引用作為參數(shù)傳給getfrecount時,創(chuàng)建了臨時引用,+1.
對象引用對象
class from_obj(object): def __init__(self, to_obj): self.to_obj = to_obj b = [1,2,3] a = from_obj(b) print(id(a.to_obj)) print(id(b))
c = [b,b]
a引用對象b,c引用b兩次。
通過objgraph包(之前安裝xdot)梳理引用拓?fù)浣Y(jié)構(gòu)。
x = [1, 2, 3] y = [x, dict(key1=x)] z = [y, (x, y)] import objgraph objgraph.show_refs([z], filename='ref_topo.png')

千萬不要兩個對象相互引用或自己引用自己,形成引用環(huán)給垃圾回收機(jī)制帶來麻煩。
垃圾回收
回收引用計數(shù)為0的對象。垃圾回收時python不能做其他任務(wù),降低效率,所以不是總隨時都垃圾回收。python記錄分配對象(object allocation)和取消分配對象(object deallocation),差值高于某閾值啟動回收。
import gc print(gc.get_threshold())
(700,10,10)700為啟動閾值,兩個10是分代回收相關(guān)閾值,通過set_threshold()重設(shè)。手動啟用垃圾回收gc.collect()。
分代回收
基本假設(shè):存活越久越不可能變垃圾。
對象分0,1,2三代。新建對象0代,經(jīng)過一次垃圾回收依然存活歸為下一代。垃圾回收啟動時一定掃描所有0代,如果0代經(jīng)過一定次數(shù)垃圾回收,則掃0和1代,同理掃0,1和2代。(700,10,10)表明10次0代配合1次1代,10次1代配合1次2代。
孤立的引用環(huán)

原本兩表對象相互引用后刪除引用,但對象引用計數(shù)部位0不會被回收。
則python復(fù)制每個對象引用計數(shù),對某個對象i,遍歷所有對象i引用的對象j,將gc_ref_j減1.
浙公網(wǎng)安備 33010602011771號