• <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>
            隨筆 - 119  文章 - 290  trackbacks - 0

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

            常用鏈接

            留言簿(12)

            隨筆分類

            我的博客

            搜索

            •  

            積分與排名

            • 積分 - 303611
            • 排名 - 84

            最新評論

            閱讀排行榜

            先說下背景,有同學看了我寫的yfgc庫的分析,留言問我能不能用c實現一個教學用gc庫,力求簡單,無需優化。這個月實在寫不出blog,正好充數一篇。
            相關連接: http://www.shnenglu.com/darkdestiny/archive/2008/09.html

            真正能用的gc庫我不會寫,簡單表達問題本質的東西倒還能應付,就寫了一下。不過只用c沒有容器是不行的,用了stl,相信同學能看明白。
            首先還是先說一下基本概念,gc本質就是把你分配出去的內存都記好了,倆倆內存之間的依賴關系都記好了。回收的時候查找依賴關系,將被根節點依賴的內存都標記為不可刪除,刪除沒有被標記的內存即可。
            說實話,gc對于c/cpp這種中級語言意義不大,因為你不得不手動修改兩塊內存之間的依賴關系,這和手動管理內存是沒有差別的。你忘了解除依賴關系,等價于忘了釋放內存。
            gc這種東西,只對于能把指針變量進行特殊編譯的編譯器,或者基于虛擬機的語言有價值。我用c/cpp的話還是傾向于手動管理內存,不管依賴關系如何復雜,只要所有權內保證唯一就可單點手動釋放;對于所有權不唯一的依賴關系,則采用引用計數機制。

            話說太多了,貼點代碼吧,也許改改就能編譯過去。
            #define ROOT (void*)NULL

            void* gc_malloc(size_t s);
            void gc_link(void* parent, void* child);
            void gc_unlink(void* parent, void* child);
            void gc_collect();


            #include 
            <map>
            static std::multimap<void*,void*> s_links, s_linkClean;
            static std::map<void*,int> s_allPtrs;
            static int s_color = 0;


            void* gc_malloc(size_t s)
            {
                
            void* ptr = ::malloc(s);
                s_allPtrs[ptr] 
            = s_color;
                
            return ptr;
            }

            void gc_link(void* parent, void* child)
            {
                s_links.insert(std::make_pair(parent,child));
            }

            void gc_unlink(void* parent, void* child)
            {
                s_links.erase(std::make_pair(parent, child));
            }

            void gc_collect()
            {
                s_color 
            = (s_color+1% 2;
                gc_mark_r(ROOT);
                s_links 
            = s_linkClean;
                s_linkClean.clear();

                auto iter 
            = s_allPtrs.begin();
                
            while(iter!=s_allPtrs.end())
                {
                    auto cur 
            = iter++;
                    
            if(cur->second != s_color)
                    {
                        ::free(cur
            ->first);
                        s_allPtrs.erase(cur);
                    }
                }
            }

            void gc_mark_r(void* parent)
            {
                
            if(parent!=ROOT && s_allPtrs[parent]==s_color)
                {
                    
            return;
                }
                s_allPtrs[parent] 
            = s_color;

                auto iter 
            = s_links.lower_bound(parent);
                auto end 
            = s_links.upper_bound(parent);
                
            for(; iter!=end; ++iter)
                {
                    s_linkClean.insert(iter);
                    
            void* child = iter->second;
                    gc_mark_r(child);
                }
            }

            int main()
            {
                
            void* p1 = gc_malloc(10);
                gc_link(ROOT, p1);
                
            void* p2 = gc_malloc(20);
                gc_link(p1, p2);
                gc_collect();
                
            return 0;
            }

            yfgc庫里的gc_enter/gc_leave是實現在gc_link/gc_unlink上的,算拓展api吧。同學完全可以自己實現的,hint是,為每個函數調用分配一點內存,建立父子函數內存上的依賴關系,函數里分配的臨時內存和函數對應的內存也建立依賴關系,函數退出時解除相關的依賴關系。

            云風不久前又給出一個基于繼承的gc解決方案,很簡單,可以看看:
            http://blog.codingnow.com/2010/02/cpp_gc.html
            posted on 2010-04-29 21:13 LOGOS 閱讀(2942) 評論(6)  編輯 收藏 引用 所屬分類: 垃圾收集

            FeedBack:
            # re: gc庫概念簡化版 2010-04-30 09:42 Kevin Lynx
            - - 果然需要自己“改改才能編譯過去”。。  回復  更多評論
              
            # re: gc庫概念簡化版 2010-04-30 10:23 Kevin Lynx
            每一次調用gc_collect的時候,s_color變為對立值(0->1, 1->0),然后gc_mark_r將位于s_links中的指針全部標記為當前的s_color值,那么在gc_collect之前gc_unlink的指針依然為原來的s_color,即未被標記,然后gc_collect回收這些未被標記的指針(指向的內存)。

            不是很明白,s_linkClean在這里的作用。  回復  更多評論
              
            # re: gc庫概念簡化版 2010-04-30 10:35 LOGOS
            @Kevin Lynx
            假設links中有這些值
            root->a,a->b,b->c
            當root->a被刪除后,links的值為
            a->b,b->c
            結果a,b,c都被回收了,而這些鏈接關系都是無效的關系,需要清理
            s_linkClean用于記錄有效的鏈接關系  回復  更多評論
              
            # re: gc庫概念簡化版[未登錄] 2010-04-30 11:03 houapple
            先謝謝LOGOS的回復,早上起來就看自己想看到的,心情那個愉悅啊!呵呵!
            恩,剛剛把程序仔細看完,大概明白了程序的思想。
            有兩個問題:
            1、如果我們不顯式調用 gc_link 去維護依賴關系,如何去知道哪些內存需要回收?
            2、因為大部分內存泄露都在被調用的函數中發生。如果我希望如下使用:
            main()
            {
            gc_init();
            foo();
            gc_collect();
            }
            當然foo()中內存分配函數使用gc_malloc()。應該用什么機制去實現?  回復  更多評論
              
            # re: gc庫概念簡化版 2010-04-30 11:56 LOGOS
            @houapple
            1.不調用gc_link,所有內存都會被回收
            2.
            foo()
            {
            gc_enter();
            ...
            gc_leave();
            }
            gc_enter分配一個表示函數調用的內存p,修改gc_malloc,分配出c時默認和函數調用棧頂的內存建立依賴關系即可:gc_link(p,c)
            gc_leave刪除當前函數和上一個函數的依賴關系:gc_unlink(p-1, p)  回復  更多評論
              
            # re: gc庫概念簡化版[未登錄] 2010-04-30 14:22 houapple
            第一點明白了,第二點考慮中!
            謝謝 LOGOS  回復  更多評論
              
            少妇久久久久久被弄到高潮| 久久w5ww成w人免费| 久久综合五月丁香久久激情| 亚洲精品午夜国产va久久| 久久久亚洲AV波多野结衣| 国产精品久久久久天天影视| 青草久久久国产线免观| 午夜精品久久久久久久久| 久久国产精品一国产精品金尊| 青青青青久久精品国产h| 欧美伊人久久大香线蕉综合69| 97久久香蕉国产线看观看| 日日狠狠久久偷偷色综合免费| 久久精品www人人爽人人| 区亚洲欧美一级久久精品亚洲精品成人网久久久久 | 国产真实乱对白精彩久久| 无码任你躁久久久久久老妇App| A级毛片无码久久精品免费| 蜜桃麻豆WWW久久囤产精品| 久久精品国产69国产精品亚洲| 久久精品成人欧美大片| 久久综合给合综合久久| 一本一道久久精品综合| 久久久久国产视频电影| 国产精品免费看久久久| 亚洲va久久久噜噜噜久久天堂| 中文字幕久久亚洲一区| 青青青青久久精品国产h久久精品五福影院1421 | 亚洲中文字幕无码久久2017| 久久久久国产精品嫩草影院| 人人狠狠综合久久亚洲88| 久久无码av三级| 蜜桃麻豆www久久| AAA级久久久精品无码区| 亚洲综合久久综合激情久久| 大香网伊人久久综合网2020| 办公室久久精品| 亚洲欧洲久久久精品| 久久精品人妻中文系列| 国内精品久久久久久久97牛牛| 久久婷婷激情综合色综合俺也去|