• <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>
            Lyt
            posts - 16,comments - 61,trackbacks - 0

            之前已經(jīng)介紹過垃圾收集器的工作機(jī)制了,這篇文章主要針對(duì)垃圾收集器的總體設(shè)計(jì)。

            容易看到,垃圾收集器分成兩個(gè)區(qū)域,SmallObjectHeap存放小型對(duì)象,LargeObjectHeap存放大型對(duì)象。SmallObjectHeap會(huì)進(jìn)行內(nèi)存縮并,而對(duì)LargeObjectHeap進(jìn)行內(nèi)存縮并顯然不合適,移動(dòng)大型對(duì)象會(huì)花很大代價(jià)。這里強(qiáng)調(diào)下,SamllObjectHeap和LargeObjectHeap各自是一個(gè)連續(xù)的內(nèi)存區(qū)域,其中三個(gè)分代只是做一下標(biāo)志而已。

                    class GC
                    {
                    
            private:
                        
            static const int LARGE_OBJECT_SIZE;                //大型對(duì)象最小大小
                        static const int SMALL_OBJECT_SIZE;                //小型對(duì)象最小大小

                        SmallObjectHeap
            * SmallHeap;                        //小型對(duì)象堆
                        LargeObjectHeap* LargeHeap;                        //大型對(duì)象堆

                        Pool
            <ObjectHandle> ObjectHandlePool;

                        
            bool IsLargeObject(const int size)const;        //判斷是否為大型對(duì)象

                    
            public:
                        
            void Clear();                                    //釋放GC申請所有內(nèi)存
                        ObjectHandle* Alloc(const int size);            //分配size大小的對(duì)象
                        void Collect(const int generationIndex=0);        //對(duì)LarObjectHeap和SmallObjectHeap中第0-generationIndex分代進(jìn)行垃圾收集
                        void Mark(ObjectHandle* handle);                //標(biāo)記對(duì)象handle,表示其為存活對(duì)象

            接下來詳細(xì)介紹下SmallObjectHeap。

                    class Generation            //分代
                    {
                    
            public:
                        
            int Start;                //開始位置
                        int Size;                //該分代大小
                        int AllocateIndex;        //該分代空閑內(nèi)存起始位置
                        int Free;                //該分代空閑內(nèi)存大小

                        
            void Init(const int start, const int size);
                        
            bool CanAlloc(const int size)const;            //該分代的空閑內(nèi)存是否足以分配size大小的對(duì)象
                        void AfterAlloc(const int size);            //分配size大小的對(duì)象后更新該分代信息
                    };

                    
            class SmallObjectHeap                        //小型對(duì)象堆
                    {
                    
            private:
                        
            static const int TOTAL;
                        
            int Total;                                //真實(shí)內(nèi)存區(qū)域Data的大小
                        int Free;                                //空閑內(nèi)存的大小

                        
            char* Data;                                //真實(shí)內(nèi)存區(qū)域
                        const int GenerationCount;                //分多少代
                        Generation* Generations;                //各分代的詳細(xì)信息
                        ObjectHandleContainer ObjectHandles;    //記錄所有分配出去的ObjectHandle,便于垃圾收集的時(shí)候更新信息

                    
            public:
                        
            void Clear();                                                                    //釋放該小型對(duì)象堆申請的所有內(nèi)存
                        ObjectHandle* Alloc(const int size, Pool<ObjectHandle>& ObjectHandlePool);        //分配size大小的對(duì)象
                        void Collect(const int generationIndex=0);                                        //對(duì)0-generationIndex代進(jìn)行垃圾收集
                    };

            下面我們看下之前一直提到的ObjectHandle。垃圾收集器對(duì)外提供的都是ObjectHandle,所有的工作都只能建立在ObjectHandle上而不是針對(duì)一個(gè)char*,包括標(biāo)記對(duì)象、回收內(nèi)存等。這里稍微提一下用ObjectHandle而非直接對(duì)char*進(jìn)行操作的好處。我們知道內(nèi)存縮并的時(shí)候,是需要把存活對(duì)象的內(nèi)存里的數(shù)據(jù)復(fù)制到別的地方去的,意味著對(duì)象所在地內(nèi)存區(qū)域會(huì)有變動(dòng),而如果這里的垃圾收集器我并不希望有內(nèi)存縮并這個(gè)動(dòng)作,這意味著對(duì)象真實(shí)存在的內(nèi)存區(qū)域并不會(huì)改變,于是char*是死的,并不會(huì)跑,如果我一律都用ObjectHandle.GetPointer()來獲得對(duì)象真實(shí)的內(nèi)存區(qū)域,那么一切文章都可以封裝在ObjectHandle里,而沒有必要垃圾收集機(jī)制的改變就大幅度地變動(dòng)代碼。

                    enum ObjectHandleType                   //區(qū)別對(duì)象是否被外部指針引用
                    {
                        handleNORMAL,
                        handlePINNED                              
            //外部指針指向的對(duì)象不可被收集
                    };

                    
            class ObjectHandle
                    {
                    
            private:
                        
            char* Data;                                    //內(nèi)存區(qū)域
                    public:
                        ObjectHandleType Type;                
            //Handle類型

                        
            int Start;                                        //開始位置
                        int Size;                                         //對(duì)象大小
                        bool Marked;                                 //對(duì)象是否被標(biāo)記

                        
            void Init(char* data, const int start, const int size, const ObjectHandleType type=handleNORMAL);
                        
            void Move(const int index);            //將對(duì)象移動(dòng)到指定的位置,參數(shù)為開始位置
                        char* GetPointer();                         //返回對(duì)象所在地內(nèi)存區(qū)域,即在Data的基礎(chǔ)上后移Start個(gè)位置
                    };
            posted on 2010-05-14 14:44 Lyt 閱讀(1847) 評(píng)論(2)  編輯 收藏 引用 所屬分類: 垃圾收集器

            FeedBack:
            # re: 稚嫩版垃圾收集器 之 具體實(shí)現(xiàn)(一)
            2010-05-14 16:04 | 陳梓瀚(vczh)
            萬一你的generation不夠大,決定再次申請一個(gè)更大的Data的時(shí)候,你的Handle里面的Data豈不是變不了了?你應(yīng)該把一個(gè)Heap的指針放進(jìn)去才是。  回復(fù)  更多評(píng)論
              
            # re: 稚嫩版垃圾收集器 之 具體實(shí)現(xiàn)(一)
            2010-05-14 16:39 | Lyt
            @陳梓瀚(vczh)
            Generation不夠大就開始垃圾收集了,把存活對(duì)象提升到更高的Generation,如果內(nèi)存還是不夠,我就拋出異常了。
            Geneation的大小一開始就折騰成固定的,不知道要根據(jù)什么規(guī)律把它弄成活的才合適。
            你的意思是讓我把Heap指針放到Generation里?  回復(fù)  更多評(píng)論
              
            国产精品欧美亚洲韩国日本久久| 精品久久久久久久| 久久天天躁夜夜躁狠狠躁2022 | 久久香综合精品久久伊人| 狼狼综合久久久久综合网| 久久福利青草精品资源站| 久久久久香蕉视频| 久久国产精品99国产精| 国产精品99久久不卡| 人妻少妇久久中文字幕一区二区| 品成人欧美大片久久国产欧美| 精品久久久久久无码不卡| 欧美伊香蕉久久综合类网站| 亚洲精品午夜国产va久久| 国产精品久久久久久福利漫画| 久久五月精品中文字幕| 99久久国语露脸精品国产| 久久精品国产亚洲AV影院| 国产成人精品久久亚洲| 2021久久国自产拍精品| 久久综合亚洲鲁鲁五月天| 久久99国产精品成人欧美| 99国产欧美久久久精品蜜芽| yy6080久久| 色婷婷久久综合中文久久一本| 国产精品久久免费| 九九精品99久久久香蕉| 久久久高清免费视频| 久久福利片| 国产精品无码久久综合网| 久久99精品国产一区二区三区| 色偷偷偷久久伊人大杳蕉| 日韩精品久久无码中文字幕| 思思久久99热只有频精品66| 日产久久强奸免费的看| 久久婷婷色综合一区二区| 狠狠色综合网站久久久久久久| 99久久精品国产一区二区三区| 亚洲狠狠综合久久| 狠狠人妻久久久久久综合| 久久久久国产精品麻豆AR影院 |