青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

隨筆 - 119  文章 - 290  trackbacks - 0

博客搬家了哦,請移步
叫我abc

常用鏈接

留言簿(12)

隨筆分類

我的博客

搜索

  •  

積分與排名

  • 積分 - 306720
  • 排名 - 84

最新評論

閱讀排行榜

終于可以看看這個gc庫是如何收集垃圾內存的了。還是老方法,先貼代碼,來一個直觀上的認識

 1void
 2gc_collect()
 3{
 4    int i;
 5    stack_pack();
 6    cache_flush();
 7    gc_mark(0);
 8    for (i=0;i<E.size;i++{}
 9    E.mark+=2;
10}
首先是stack_pack,這將是要研究的第一個子函數。根據其名字,可以推測會操作前面看過的管理自由內存的堆棧,事實上也是如此。該函數將清理堆棧,將暫時不能釋放的自由內存用一種自動化的方式標記出來,這種方式就是為他們建立依賴關系。建立依賴關系也是內存不能被當作垃圾的唯一理由。
下一步是cache_flush,這個以前已經看過,將cache的依賴關系寫入到實際管理的容器上。
第三步是gc_mark,用來給所有有依賴關系的內存做一下標記,證明他們不是垃圾。
第四步是一個for循環,循環體太長暫時先不貼出來,不過其工作就是將標記不合格的內存釋放,因為他們是垃圾內存。
最后更新標記,標記是一個遞增的整數。

垃圾收集的過程直觀上就是這樣,現在就來逐步細看,先看看stack_pack。聲明一點,通過這個函數,可以解開之前在gc_leave中為何會有E.bottom大于 E.current的緣故(除了初始化)。
1static void
2stack_pack()
3{
4    int bottom=stack_pack_internal(E.stack.bottom,E.stack.current,E.stack.top);
5    E.stack.top=E.stack.bottom=bottom;
6    E.stack.current=bottom-1;
7}
stack_pack的代碼很少,因此可以斷定關鍵的地方都在stack_pack_internal上了,其返回值bottom也有重要意義。
第5、6行,直接將top,bottom,current指針重置到一個類似于初始化的狀態,不過此時的bottom通常都已經往上移動了,而current小于bottom。
也就是說,每次調用gc_collect后,總會造成current小于bottom的結果,那么gc_leave的實現里有對這種判斷的處理也就不奇怪了,不過具體這種處理是何用意,看了stack_pack_internal之后再說。

 1static int
 2stack_pack_internal(int from,int to,int top)
 3{
 4    if (to < from) {
 5        int parent = E.stack.data[to].stack;
 6        while (from < top) {
 7            node_add(parent,E.stack.data[from].handle);
 8            ++from;
 9        }

10        return to+1;
11    }

12    else {
13        int bottom=stack_pack_internal(from,to-E.stack.data[to].number,to);
14        int node=node_alloc(0);
15        ++to;
16        while (to<top) {
17            node_add(node,E.stack.data[to].handle);
18            ++to;
19        }

20        node_add(E.stack.data[bottom-1].stack,node);
21        E.stack.data[bottom].stack=node;
22        return bottom+1;
23    }

24}

快速掃一眼stack_pack_internal,發現他是一個遞歸函數,其次確認一下三個參數,分別對應bottom,current,top,也就是當前函數下的堆棧指針位置。
第4行的代碼可以看成 if ( current < bottom ),通過以前看過的代碼,可以了解到以下3點:
1.初始化之后,current < bottom
2.調用gc_collect之后,current < bottom
3.調用了gc_enter后,current 絕對不會小于bottom
因此else部分的代碼才是要先考慮的代碼,也就是遞歸的部分。

第13行的代碼可以寫成
bottom = stack_pack_internal( bottom , current - E.stack.data[ current ].number , current )
有看出什么來嗎?比如說從number,或者是從這個減法。沒錯喲,這三個用于遞歸的參數,其實是在父函數環境下的堆棧指針位置。也就是說,stack_pack_internal遞歸的調用,由于執行常規的else部分的代碼,因此不斷的傳遞父函數的堆棧指針位置,不斷的尋根,不斷的追宗認祖,最終就會回到stack剛初始化的狀態,此時 current < bottom。很酷的,現在不得不暫時放下else部分,來看看if部分,這if部分可就只是為了這么一下而準備的。
第5行,取的是current節點處的值。剛初始化的stack,這里存放著全局內存之根,最大的所有者。
接下來,由于此時bottom 等于 top,所以while循環暫不執行,直接返回bottom。現在可以回到else部分了。

第14行,從 E.pool中分配了一個節點,當然該節點實際沒有維護任何內存,不過這個節點可有大用處。
第16到第19行,這個while循環內,將某一級函數(遞歸太多,是哪一級已經不重要了)中分配出來的自由內存和剛申請的node建立依賴關系,這級函數的自由內存id可都記錄在current+1到top之間的handler中呢(current記錄父函數分配的自由內存數量)。
第20行,將剛申請node和 E.stack.data[ bottom -1 ].stack建立以來關系,不過這到底是什么呢?如果bottom-1等于0的話,那就是node和全局建立依賴關系,否則,看起來就像是和父函數建立了依賴關系。第21行驗證了這個想法,這一賦值,將node記錄到了堆棧中。就此也知道了stack成員變量的作用,表示某個函數的節點。
最后返回bottom+1,即bottom指針移動了,而bottom-1必然指向了代表父函數的node。

以上的分析看其來都比較混亂,不過從整體想像的話:
假設函數其實是一種對象,他引用著那些在他函數體內分配出來的自由內存,因此他和那些內存就有依賴關系。從最上層的函數開始,建立表示該函數對象的節點,和所有在這一級函數中分配的自由內存建立依賴關系。然后到下一級子函數,也建立一個表示該函數對象的節點,和所有在這一級函數中分配的自由內存建立依賴關系。如此遞歸,直到調用了gc_collect的這一級函數為止。
管理自由內存的堆棧,經過這樣處理后,里面就剩下了每一級函數對象的節點id,一個個的緊挨著。而作為到調用gc_collect為止,這一路下來分配的自由內存,都是暫時不能釋放的,已經建立依賴關系放進了 E.cache。

根據stack_pack后兩行的代碼,就可以判斷出if代碼中while循環是用來干什么吃的了:同一級函數中,在gc_collect后,繼續分配自由內存,然后再gc_collect的話,就會執行到該while循環,本質也是用來建立依賴關系的。


最后說說看gc_leave時無法理解的代碼吧,現在已是真相大白了
 1    else {
 2        int parent,child;
 3        --E.stack.bottom;
 4        parent=E.stack.data[E.stack.bottom-1].stack;
 5        child=E.stack.data[E.stack.bottom].stack;
 6        node_add(parent, child | UNSET_MASK);
 7        node_free(child);
 8        E.stack.current=E.stack.bottom-1;
 9        E.stack.top = E.stack.current + 1;
10    }
這是current小于bottom的情形,已經被gc_collect了,此時 E.stack.data[0]到 bottom指針之間應該都是每一級函數的象征節點。
第4、5行,就是一個父函數和子函數的關系,現在既然已經從子函數中退出了,那么也是時候解放子函數中分配的自由內存的時候了,因此第6行解開了父子函數的依賴關系,那子函數中分配的自由內存也相應的變成垃圾了。
最后兩行恢復父函數的堆棧,我想堆棧的形狀也差不多了,E.stack.data[0]到 bottom指針之間應該都是每一級函數的象征節點。
posted on 2008-09-21 23:17 LOGOS 閱讀(1696) 評論(3)  編輯 收藏 引用

FeedBack:
# re: 垃圾收集的那點事(I) 2008-09-22 11:38 來支持
天天來支持下。
博主有沒有看過一個autofreealloc的玩意?  回復  更多評論
  
# re: 垃圾收集的那點事(I) 2008-09-22 14:11 LOGOS
@來支持
沒聽說過,搜索了一下,不知道你說的是不是這個
http://blog.csdn.net/xushiweizh/archive/2006/11/19/1396573.aspx

這個autofreealloc的責任很明確-----理解該垃圾回收器的關鍵點在于,是在于理解它的目標:為一個復雜的局部過程(算法)提供自動內存回收的能力。
所以從各種意義上他都比yfgc簡單得多  回復  更多評論
  
# re: 垃圾收集的那點事(I) 2008-09-22 22:40 來支持
是的。
許這個所謂半自動GC花了一些時間研究過他的實現,感覺應用面比較小,而且只適合單線程環境。爭議也比較大

個人對C/C++下輕量GC實現非常敢興趣,云風這個還沒時間深入去看,現在甚至還不知道它能做什么,不能做什么,在什么環境下適用,什么環境下不適用。

提個建議,博主你能不能對云風這個GC寫一個綜述性和應用實例的文章,然后再去探討內部的實現機理。單純的代碼分析好像C++博客的弟兄們都不怎么感冒,這么好的東西都沒人關注。呵呵。



  回復  更多評論
  

只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
      <noscript id="pjuwb"></noscript>
            <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
              <dd id="pjuwb"></dd>
              <abbr id="pjuwb"></abbr>
              韩国一区二区三区在线观看| 亚洲国产一区二区a毛片| 久久精品99国产精品| 一区精品在线| 亚洲激情电影在线| 欧美日韩在线精品| 欧美一区激情视频在线观看| 久久精品视频99| 日韩亚洲成人av在线| 亚洲午夜女主播在线直播| 国产无遮挡一区二区三区毛片日本| 久久综合一区二区| 欧美理论在线| 欧美一区二区成人| 老司机精品久久| 亚洲视频在线免费观看| 亚洲欧美在线一区| 亚洲激情婷婷| 亚洲素人在线| 亚洲第一中文字幕| 一本色道久久88精品综合| 国产亚洲一区二区三区在线播放| 蜜桃av综合| 国产精品hd| 欧美~级网站不卡| 欧美午夜精彩| 你懂的一区二区| 欧美手机在线| 嫩模写真一区二区三区三州| 欧美三级在线视频| 另类人畜视频在线| 国产精品国产三级国产普通话三级 | 老牛影视一区二区三区| 欧美日韩另类在线| 久久一综合视频| 欧美日韩在线播放| 男女精品视频| 国产精品一二三视频| 亚洲国产第一| 国产亚洲免费的视频看| 亚洲精品欧洲| 国模精品一区二区三区色天香| 91久久久久久久久久久久久| 国产毛片一区二区| 亚洲欧洲一区二区在线观看| 国产一级一区二区| 99re66热这里只有精品3直播| 国产三级欧美三级日产三级99| 亚洲国产精品成人综合色在线婷婷 | 欧美精品日韩一区| 久久天堂av综合合色| 欧美午夜免费影院| 亚洲第一区在线| 国产亚洲毛片在线| 在线视频一区观看| 亚洲每日更新| 久久综合九色| 久久久久网址| 国产精品一区视频网站| 亚洲精一区二区三区| 亚洲国产午夜| 久久精品国产亚洲aⅴ| 午夜亚洲激情| 欧美日韩亚洲一区二区三区| 欧美国产另类| 激情成人中文字幕| 亚洲欧美日韩系列| 亚洲综合精品四区| 欧美日韩免费观看一区=区三区| 欧美va亚洲va日韩∨a综合色| 国产亚洲成av人在线观看导航| 一本一本a久久| 一本色道久久综合亚洲二区三区| 另类欧美日韩国产在线| 久久综合福利| 激情伊人五月天久久综合| 亚洲一区三区在线观看| 亚洲午夜av电影| 欧美—级a级欧美特级ar全黄| 免费观看一级特黄欧美大片| 国产亚洲欧美在线| 亚洲欧美日韩人成在线播放| 亚洲一区影音先锋| 欧美日韩视频免费播放| 亚洲黄色一区二区三区| 最新成人av网站| 另类亚洲自拍| 欧美88av| 亚洲国产日韩欧美综合久久| 久久精品国产精品亚洲综合| 久久国产精品久久国产精品| 国产精品欧美在线| 亚洲一区欧美激情| 欧美在线观看视频在线| 国产精品一区久久久| 亚洲一区二区精品在线| 亚洲欧美自拍偷拍| 国产精品女主播| 亚洲欧美国产另类| 欧美专区一区二区三区| 国产日韩综合| 欧美一区中文字幕| 久久综合色婷婷| 在线国产精品播放| 免费国产自线拍一欧美视频| 亚洲成人在线视频播放 | 欧美一级午夜免费电影| 久久黄色影院| 狠狠干综合网| 裸体丰满少妇做受久久99精品| 欧美成熟视频| 亚洲免费观看视频| 欧美日韩视频一区二区| 在线亚洲+欧美+日本专区| 亚洲欧美日本伦理| 国产日韩一区二区三区在线播放| 久久国产加勒比精品无码| 裸体丰满少妇做受久久99精品 | 久久国产一区| 免费在线欧美视频| 亚洲精品在线视频| 欧美日韩在线播| 亚洲男人影院| 乱人伦精品视频在线观看| 亚洲韩国日本中文字幕| 欧美极品在线视频| 亚洲美女在线观看| 亚洲欧美日韩国产另类专区| 国产日韩一区二区| 玖玖精品视频| 亚洲精品乱码久久久久久日本蜜臀| 中文欧美字幕免费| 国产日本欧美在线观看| 久久婷婷麻豆| 日韩一级视频免费观看在线| 性色一区二区三区| 在线精品一区| 欧美三级电影一区| 欧美一区二区三区四区在线 | 欧美自拍丝袜亚洲| 欧美激情中文字幕在线| 亚洲午夜视频在线观看| 国产丝袜美腿一区二区三区| 久久久综合网站| 亚洲美女黄网| 欧美在线观看一区二区| 1769国内精品视频在线播放| 欧美日韩久久久久久| 先锋亚洲精品| 亚洲国产精选| 久久国产精品久久精品国产| 91久久亚洲| 国产精品一区二区三区乱码 | 欧美精品久久久久久久免费观看 | 欧美精品观看| 亚洲欧美国产精品va在线观看| 欧美不卡在线| 中日韩视频在线观看| 国产在线观看91精品一区| 欧美激情一区二区三区蜜桃视频 | 久久精品综合| 99精品国产99久久久久久福利| 久久精品网址| 一区二区三区导航| 激情五月综合色婷婷一区二区| 欧美日韩午夜在线| 久久久久**毛片大全| 一区二区精品国产| 免费观看欧美在线视频的网站| 亚洲天堂成人在线视频| 亚洲大片一区二区三区| 欧美激情视频一区二区三区在线播放 | 国产精品夜色7777狼人 | 欧美日韩直播| 久久久久久电影| 亚洲一区视频| 亚洲精品日韩在线| 久久野战av| 亚洲欧美日韩国产成人| 亚洲人妖在线| 国产手机视频一区二区| 欧美精品九九| 久久久精品999| 一区二区免费看| 亚洲国产精品一区二区久 | 国产三级欧美三级日产三级99| 欧美人成免费网站| 久久全球大尺度高清视频| 日韩亚洲一区二区| 免费成人美女女| 香蕉av777xxx色综合一区| 夜夜嗨av一区二区三区四季av | 欧美三级午夜理伦三级中文幕| 久久影院午夜片一区| 亚洲欧美一区二区激情| 一卡二卡3卡四卡高清精品视频| 亚洲国产精品久久精品怡红院| 久久亚洲综合| 久久久国产亚洲精品| 亚洲欧美亚洲|