亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

基于malloc與free函數(shù)的實現(xiàn)代碼及分析

 更新時間:2013年05月04日 09:38:54   作者:  
本篇文章介紹了malloc與free函數(shù)的實現(xiàn)代碼及分析。需要的朋友參考下

  用于內存管理的malloc與free這對函數(shù),對于使用C語言的程序員應該很熟悉。前段時間聽說有的IT公司以“實現(xiàn)一個簡單功能的malloc”作為面試題,正好最近在復習K&R,上面有所介紹,因此花了些時間仔細研究了一下。畢竟把題目做出來是次要的,了解實現(xiàn)思想、提升技術才是主要的。本文主要是對malloc與free實現(xiàn)思路的介紹,藍色部分文字是在個人思考中覺得比較核心的東西;另外對于代碼的說明,有一些K&R上的解釋,使用綠色加亮。

  在研究K&R第八章第五節(jié)的實現(xiàn)之前,不妨先看看其第五章第四節(jié)的alloc/afree實現(xiàn),雖然這段代碼主要目的是展示地址運算。

復制代碼 代碼如下:

alloc實現(xiàn)

#define ALLOCSIZE 10000
static char allocbuf[ALLOCSIZE];    /*storage for alloc*/
static char *allocp = allocbuf;    /*next free position*/

char *alloc(int n)
{
    if(allocbuf+ALLOCSIZE - allocp >= n) {
        allocp += n;
        return alloc - n;
    } else
        return 0;
}

void afree(char *p)
{
    if (p >= allocbuf && p<allocbuf + ALLOCSIZE)
        allocp = p;
}

  這種簡單實現(xiàn)的缺點

    1.作為代表內存資源的allocbuf,其實是預先分配好的,可能存在浪費。

    2.分配和釋放的順序類似于棧,即“后進先出”,釋放時如果不按順序會造成異常。

  這個實現(xiàn)雖然比較簡陋,但是依然提供了一個思路。如果能把這兩個缺點消除,就能夠實現(xiàn)比較理想的malloc/free。

  僅僅依靠地址運算來進行定位,是限制分配回收靈活性的原因,它要求已使用部分和未使用部分必須通過某個地址分開成兩個相鄰區(qū)域。為了能讓這兩個區(qū)域能夠互相交錯,甚至其中還包括一些沒有分配的地址空間,需要使用指針把同類的內存空間連接起來形成鏈表,這樣就可以處理地址不連續(xù)的一系列內存空間。但是為什么只連接了空閑空間而不連接使用中的空間?這么問可能出于在對圖中二者類比時的直覺而沒有經過思考,這很簡單,因為沒有必要。前者相互鏈接是為了能夠在內存分配時遍歷所有空閑空間,并且在使用free()回收已使用空間時進行重新插入。而對于使用中的空間,由于我們在分配空間時已經知道它們的地址了,回收時可以直接告訴free(),并不用像malloc()時進行遍歷。

  既然提到了鏈表,可能對數(shù)據(jù)結構稍有了解的人會立刻寫下一個struct來代表一個內存區(qū)域,其中包含一個指向下一個內存區(qū)域的指針,但是這個struct的其他成員該怎么寫呢?作為待分配的內存區(qū)域,大小是不定的,如果把它聲明為struct的成員變量顯然不妥;如果聲明為一個指向某個其他的區(qū)域的指針,這似乎又和上面的直觀表示不相符合。(當然,這么做也是可以實現(xiàn)的,它看上去是介于上圖的兩者之間,把管理結構和實際分配的空間相剝離,在文末我會專門的討論一下這種實現(xiàn)方法)因此,這里仍然把控制結構和空閑空間相分開,但保持它們在內存地址中相鄰,形成下圖的形式,而正由這個特點,我們可以利用對控制結構指針的指針運算來定位對應的內存區(qū)域:

  

  對應地,把控制信息定義為Header:

復制代碼 代碼如下:

typedef long Align;/*for alignment to long boundary*/
union header {
    struct {
        union header *ptr; /*next block if on free list*/
        unsigned size; /*size of this block*/
    } s;
    Align x;
};

typedef union header Header;

  使用union而不是直接使用struct的原因是為了地址對齊。這里是long對齊,union的x永遠不會使用。

  這樣,malloc的主要工作就是對這些Header和其后的內存塊的管理。

復制代碼 代碼如下:

malloc()

static Header base;
static Header *freep = NULL;

void *malloc(unsigned nbytes)
{
    Header *p, *prevp;
    unsigned nunits;
    nunits = (nbytes+sizeof(Header)-1)/sizeof(Header) + 1;
    if((prevp = freep) == NULL) { /* no free list */
        base.s.ptr = freep = prevp = &base;
        base.s.size = 0;
    }
    for(p = prevp->s.ptr; ;prevp = p, p= p->s.ptr) {
        if(p->s.size >= nunits) { /* big enough */
            if (p->s.size == nunits)  /* exactly */
                prevp->s.ptr = p->s.ptr;
            else {
                p->s.size -= nunits;
                p += p->s.size;
                p->s.size = nunits;
            }
            freep = prevp;
            return (void*)(p+1);
        }
        if (p== freep) /* wrapped around free list */
            if ((p = morecore(nunits)) == NULL)
                return NULL; /* none left */
    }
}


  實際分配的空間是Header大小的整數(shù)倍,并且多出一個Header大小的空間用于放置Header。但是直觀來看這并不是nunits = (nbytes+sizeof(Header)-1)/sizeof(Header) + 1啊?如果用(nbytes+sizeof(Header))/sizeof(Header)+1豈不是剛好?其實不是這樣,如果使用后者,nbytes+sizeof(Header)%sizeof(Header) == 0時,又多分配了一個Header大小的空間了,因此還要在小括號里減去1,這時才能符合要求。

   malloc()第一次調用時建立一個退化鏈表base,只有一個大小是0的空間,并指向它自己。freep用于標識空閑鏈表的某個元素,每次查找時可能發(fā)生變化;中間的查找和分配過程是基本的鏈表操作,在空閑鏈表中不存在合適大小的空閑空間時調用morecore()獲得更多內存空間;最后的返回值是空閑空間的首地址,即Header之后的地址,這個接口與庫函數(shù)一致。

復制代碼 代碼如下:

morecore()

#define NALLOC 1024    /* minimum #units to request */
static Header *morecore(unsigned nu)
{
    char *cp;
    Header *up;
    if(nu < NALLOC)
        nu = NALLOC;
    cp = sbrk(nu * sizeof(Header));
    if(cp == (char *)-1)    /* no space at all*/
        return NULL;
    up = (Header *)cp;
    up->s.size = nu;
    free((void *)(up+1));
    return freep;
}


  morecore()從系統(tǒng)申請更多的可用空間,并加入。由于調用了sbrk(),系統(tǒng)開銷比較大,為避免morecore()本身的調用次數(shù),設定了一個NALLOC,如果每次申請的空間小于NALLOC,就申請NALLOC大小的空間,使得后續(xù)malloc()不必每次都需要調用morecore()。對于sbrk(),在后面會有介紹。

  這里有個讓人驚訝的地方:malloc()調用了morecore(),morecore()又調用了free()!第一次看到這里時可能會覺得不可思議,因為按照慣性思維,malloc()和free()似乎應該是相互分開的,各司其職???但請再思考一下,free()是把空閑鏈表進行擴充,而malloc()在空閑鏈表不足時,從系統(tǒng)申請到更多內存空間后,也要先把它們轉化成空閑鏈表的一部分,再進行利用。這樣,malloc()調用free()完成后面的工作也是順理成章了。根據(jù)這個思想,后面是free()的實現(xiàn)。在此之前,還有幾個morecore()自身的細節(jié):

  1.如果系統(tǒng)也沒有空間可以分配,sbrk()返回-1。cp是char *類型,在有的機器上char無符號,這里需要一次強制類型轉換。

  2.morecore()調用的返回值看上去比較奇怪,別擔心,freep會在free()中修改的。使用這個返回值也是為了在malloc()里的判斷、p = freep的再次賦值的語句能夠緊湊。

復制代碼 代碼如下:

free()

void free(void *ap)
{
    Header *bp,*p;
    bp = (Header *)ap -1; /* point to block header */
    for(p=freep;!(bp>p && bp< p->s.ptr);p=p->s.ptr)
        if(p>=p->s.ptr && (bp>p || bp<p->s.ptr))
            break;    /* freed block at start or end of arena*/
    if (bp+bp->s.size==p->s.ptr) {    /* join to upper nbr */
        bp->s.size += p->s.ptr->s.size;
        bp->s.ptr = p->s.ptr->s.ptr;
    } else
        bp->s.ptr = p->s.ptr;
    if (p+p->s.size == bp) {     /* join to lower nbr */
        p->s.size += bp->s.size;
        p->s.ptr = bp->s.ptr;
    } else
        p->s.ptr = bp;
    freep = p;
}


   free()首先定位要釋放的ap對應的bp與空閑鏈表的相對位置,找到它的的最近的上一個和下一個空閑空間,或是當它在整個空閑空間的前面或后面時找到空閑鏈表的首尾元素。注意,由于malloc()的分配方式和free()的回收時的合并方式(下文馬上要提到),可以保證整個空閑空間的鏈表總是從低地址逐個升高,在最高地址的空閑空間回指向低地址第一個空閑空間。

  定位后,根據(jù)要釋放的空間與附近空間的相鄰性,進行合并,也即修改對應空間的Header。兩個if并列可以使得bp可以同時與高地址和低地址空閑空間結合(如果都相鄰),或者進行二者之一的合并,或者不合并。

  完成了這三部分代碼后(注意放到同一源文件中,sbrk()需要#include <unistd.h>),就可以使用了。當然要注意,命名和stdlib.h中的同名函數(shù)是沖突的,可以自行改名。

  第一次審視源碼,會發(fā)現(xiàn)很多實現(xiàn)可能原先并沒有想到:Header的結構和對齊填充、空間的取整、鏈表的操作和初始化(邊界情況)、malloc()對free()的調用、由malloc()和free()暗中保證的鏈表地址有序等等,確實很值得玩味。另外再附上前文中提到的兩個問題還有一些補充問題的簡單思考

1.Header與空閑空間相剝離,Header中包含一個指向其空閑空間的指針

  這樣做未必不可,相應地算法需要改動。同時,由于Header和空閑空間不再相鄰,sbrk()獲得的空間也應該包含Header的部分,內存的分布可能會更加瑣碎。當然,這也可能帶來好處,即用其他數(shù)據(jù)結構對鏈表進行管理,比如按大小進行hash,這樣查找起來更快。

2.關于sbrk()

  sbrk()也是庫函數(shù),它能使堆往棧的方向增長,具體可以參考:brk(), sbrk() 用法詳解。

3.可以改進的方

  空閑空間的尋找是線性的,查找過程在內存分配中可以看作是循環(huán)首次適應算法,在某些情況下可能很慢;如果再建立一個數(shù)據(jù)結構,如hash表,對不同大小的空間進行索引,肯定可以加快查找本身,并且能實現(xiàn)一些算法,比如最佳匹配。但查找加快的代價是,修改這個索引會占用額外的時間,這是需要權衡的。

  morecore()中的最小分配空間是宏定義,在實際使用中完全可以作為參數(shù)傳遞,根據(jù)需要設定最小分配下限。

相關文章

  • Qt?QString的使用實現(xiàn)

    Qt?QString的使用實現(xiàn)

    本文主要介紹了Qt?QString的使用實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2022-05-05
  • C語言實現(xiàn)Floyd算法

    C語言實現(xiàn)Floyd算法

    這篇文章主要為大家詳細介紹了C語言實現(xiàn)Floyd算法,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-01-01
  • C++設計模式之備忘錄模式(Memento)

    C++設計模式之備忘錄模式(Memento)

    這篇文章主要為大家詳細介紹了C++設計模式之備忘錄模式Memento的相關資料,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-04-04
  • C語言strlen函數(shù)實現(xiàn)讀取字符串長度詳解

    C語言strlen函數(shù)實現(xiàn)讀取字符串長度詳解

    這篇文章主要介紹了用C語言的strlen函數(shù)來實現(xiàn)讀取字符串長度的過程,strlen所作的是一個計數(shù)器的工作,它從內存的某個位置開始掃描,直到碰到第一個字符串結束符'\0'為止
    2022-04-04
  • C語言+MySQL實現(xiàn)推箱子游戲

    C語言+MySQL實現(xiàn)推箱子游戲

    經典的推箱子是一個來自日本的古老游戲,目的是在訓練你的邏輯思考能力。本文將通過C語言和MySQL實現(xiàn)推箱子這一經典游戲,感興趣的可以了解一下
    2022-02-02
  • C++實現(xiàn)LeetCode(98.驗證二叉搜索樹)

    C++實現(xiàn)LeetCode(98.驗證二叉搜索樹)

    這篇文章主要介紹了C++實現(xiàn)LeetCode(98.驗證二叉搜索樹),本篇文章通過簡要的案例,講解了該項技術的了解與使用,以下就是詳細內容,需要的朋友可以參考下
    2021-07-07
  • C++ 關于MFC多線程編程的注意事項

    C++ 關于MFC多線程編程的注意事項

    這篇文章主要介紹了C++ 關于MFC多線程編程的注意事項的相關資料,需要的朋友可以參考下
    2015-06-06
  • 利用C語言玩轉魔方陣實例教程

    利用C語言玩轉魔方陣實例教程

    這篇文章主要給大家介紹了關于利用C語言玩轉魔方陣的相關資料,文中詳細介紹了關于奇數(shù)魔方陣和4N 魔方陣的實現(xiàn)方法,通過示例代碼讓大家更好的參考學習,需要的朋友們下面隨著小編來一起學習學習吧。
    2017-11-11
  • C++11之后的decltype類型指示符詳解

    C++11之后的decltype類型指示符詳解

    為了滿足這一要求,C++11?新標準引入了另一種類型說明符?decltype?,它的作用是選擇并返回操作數(shù)的數(shù)據(jù)類型,這篇文章主要介紹了C++11之后的decltype類型指示符,需要的朋友可以參考下
    2023-01-01
  • C語言自定義函數(shù)的實現(xiàn)

    C語言自定義函數(shù)的實現(xiàn)

    這篇文章主要介紹了C語言自定義函數(shù)的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-01-01

最新評論