分代垃圾回收,基于的是“大部分的對(duì)象,在生成后馬上就會(huì)變成垃圾”這一經(jīng)驗(yàn)上的事實(shí)為設(shè)計(jì)出發(fā)點(diǎn)。此前討論過(guò)基于引事實(shí)的另一個(gè)垃圾回收算法,引用計(jì)數(shù)出的一些優(yōu)化思路。

 

分代的關(guān)鍵是:

  1. 給對(duì)象記錄下一個(gè)age,隨著每一次垃圾回收,這個(gè)age會(huì)增加;

  2. 給不同age的對(duì)象分配不同的堆內(nèi)內(nèi)存空間,稱(chēng)為某一代;

  3. 對(duì)某一代的空間,有適合其的垃圾回收算法;

  4. 對(duì)每代進(jìn)行不同垃圾回收,一般會(huì)需要一個(gè)額外的信息:即每代中對(duì)象被其他代中對(duì)象引用的信息。這個(gè)引用信息對(duì)于當(dāng)前代來(lái)說(shuō),扮演與"root"一樣的角色,也是本代垃圾回收的起點(diǎn)。

 

 

分代垃圾回收的典型是Ungar的分代垃圾回收。

它將堆分成如下形式:

seo優(yōu)化培訓(xùn),網(wǎng)絡(luò)推廣培訓(xùn),網(wǎng)絡(luò)營(yíng)銷(xiāo)培訓(xùn),SEM培訓(xùn),網(wǎng)絡(luò)優(yōu)化,在線營(yíng)銷(xiāo)培訓(xùn)

 

如上,分成新生代與老年代。

在新生代內(nèi),又分成了生成空間與幸存空間。當(dāng)生成空間滿(mǎn)了,會(huì)以復(fù)制算法進(jìn)行垃圾回收,復(fù)制到幸存空間中。和前面的復(fù)制算法匹配,幸存空間又一分為二,分成from和to空間。每次新生代的垃圾回收,會(huì)同時(shí)進(jìn)行生成空間到to、from空間到to的兩個(gè)垃圾回收。

seo優(yōu)化培訓(xùn),網(wǎng)絡(luò)推廣培訓(xùn),網(wǎng)絡(luò)營(yíng)銷(xiāo)培訓(xùn),SEM培訓(xùn),網(wǎng)絡(luò)優(yōu)化,在線營(yíng)銷(xiāo)培訓(xùn)

 

對(duì)于老年代,則直接進(jìn)行mark_sweep回收。

對(duì)于“記錄集”(record set),是記錄代間引用的一個(gè)數(shù)組。它內(nèi)部不能只記錄被引用對(duì)象,因?yàn)楸灰脤?duì)象被復(fù)制到to空間后,引用者本身的引用指針要更新,只記錄被引用的新生代內(nèi)對(duì)象是無(wú)法找到被引用者的。所以,必須在記錄集中記錄老年代內(nèi)對(duì)象。

 

更新記錄集的操作在分配新對(duì)象,并設(shè)置成老對(duì)象的一個(gè)field時(shí)進(jìn)行:

 

seo優(yōu)化培訓(xùn),網(wǎng)絡(luò)推廣培訓(xùn),網(wǎng)絡(luò)營(yíng)銷(xiāo)培訓(xùn),SEM培訓(xùn),網(wǎng)絡(luò)優(yōu)化,在線營(yíng)銷(xiāo)培訓(xùn)

write_barrier(obj, field, new_obj) {  if obj >= $old_start && new_obj < $old_start && obj.remembered == false            // 條件,很明顯
    $rs[$rs_idx++] = obj                   // 更新記錄集