国产激情自拍_国产9色视频_丁香花在线电影小说观看 _久久久久国产精品嫩草影院

首頁 > 編程 > C > 正文

基于malloc與free函數的實現代碼及分析

2020-01-26 16:18:28
字體:
來源:轉載
供稿:網友

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

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

復制代碼 代碼如下:

alloc實現

#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;
}

  這種簡單實現的缺點

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

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

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

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

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

  

  對應地,把控制信息定義為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大小的整數倍,并且多出一個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用于標識空閑鏈表的某個元素,每次查找時可能發生變化;中間的查找和分配過程是基本的鏈表操作,在空閑鏈表中不存在合適大小的空閑空間時調用morecore()獲得更多內存空間;最后的返回值是空閑空間的首地址,即Header之后的地址,這個接口與庫函數一致。

復制代碼 代碼如下:

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

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

  1.如果系統也沒有空間可以分配,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()的回收時的合并方式(下文馬上要提到),可以保證整個空閑空間的鏈表總是從低地址逐個升高,在最高地址的空閑空間回指向低地址第一個空閑空間。

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

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

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

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

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

2.關于sbrk()

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

3.可以改進的方

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

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

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表

圖片精選

国产激情自拍_国产9色视频_丁香花在线电影小说观看 _久久久久国产精品嫩草影院
97最新国自产拍视频在线完整在线看| 亚洲字幕成人中文在线观看| 开心婷婷激情五月| 免费看ww视频网站入口| 浪潮av一区| 国产精品一区二区婷婷| 国产精品久久久久久久久鸭 | 亚洲一区二区三区在线观看网站| 伊人福利在线| 国产在线麻豆精品| 影音先锋中文字幕在线| 国产中文字幕在线播放| 精品日韩av| 综合图区亚洲白拍在线| 中文乱码字幕高清在线观看| sese在线视频| 9999在线视频| 国产尤物一区二区三区| 在线中文字幕资源| 最近中文字幕mv免费高清电影| 精品网站www| 丁香六月婷婷| 日本h视频在线观看| 国产黄色片大全| 99热最新网址| 午夜视频在线免费| 久久香蕉av| 青青草免费在线视频| 在线黄色国产电影| 91久久精品国产性色| 尤物网在线观看| 丁香视频五月| 天堂在线一二区| 日本中文字幕视频| 国产真实伦在线观看| 日本福利在线观看| 在线视频福利| 2018中文字幕在线| 国产wwww| av大片在线| 亚洲成人av在线影院| 在线国产91| 在线观看的av网站| 在线国产一级| www.香蕉视频在线观看| 欧美a免费在线| 欧美精品se| 国产69精品久久app免费版 | 国产一级影片| 国产小视频在线| 国产精品国产国产aⅴ| 国产图片综合| 精品入口蜜桃| 日本调教视频在线观看| 福利视频在线导航| 国产精品入口麻豆免费| 国产aa视频| 亚洲精品影院在线| 超碰国产在线观看| 中文字幕在线影院| 国产精品久久久久久福利| 国产91在线视频蝌蚪| 在线观看国产福利视频| 青娱乐在线视频观看| 国产精品一卡二卡三卡| 992tv在线观看在线播放| 97中文字幕| 国产性网软件大全| 国产免费av网站| 狂野欧美性猛交xxxx乱大交| 国产精品自产拍在线网站| 国产在线一二三| 久久久久久久久免费视频| 麻豆精品传媒视频观看| 国产在线看片| 国产精品视频一区麻豆| 九九热视频免费观看| 国产麻豆精品入口在线观看| 丁香视频免费观看| av高清资源| 久久精品国产麻豆| 国产精品乱码一区二区三区视频| 一区二区三区免费视频网站| 狠狠干婷婷色| 亚洲日本一区二区三区在线观看| 丁香花在线电影小说观看| 精品欧美不卡一区二区在线观看| www免费在线观看| 国产精品人人爱一区二区白浆| 国产中文在线视频| 香蕉视频网站在线播放| 国产一二三区在线视频| av手机免费在线观看| 天天操人人干| eeuss在线观看| 国产福利小视频在线观看| 日本视频一二三区中文字幕| 国产精品免费麻豆入口| 国产免费av高清在线| 人成在线免费视频| 亚洲欧美自拍另类| 91看片在线观看| 国产福利图片| 国产精品bbw一区二区三区| 黄网址在线播放免费| 精品一区二区在线欧美| 国产永久免费高清在线观看视频| av小说在线| 亚洲sss视频| 四虎一区二区三区| 美女永久在线网站| 福利视频网站导航| 久久精品最新免费国产成人| 亚洲欧美日韩成人网| av超碰在线| 欧美日韩不卡中文字幕在线| 日本在线视频www鲁啊鲁| 亚洲人在线播放| eeuss影院www在线观看| 亚洲jjzzjjzz在线观看| 亚洲精品xxxxx| 精品一区二区91| 一本大道久久a久久精品| 亚洲欧美日韩成人网| 免费看的av| 国产私人尤物无码不卡| 国产成免费视频| 一级二级在线观看| 欧美日韩在线视频免费观看 | av日韩国产| av在线免费播放网站| 欧美性受xxxx免费视频| 亚洲免费国产| 国产毛片在线看| 一个人看的www免费观看视频| 不卡av免费观看| 五月天天在线| 日本成人在线播放| av片在线观看永久免费| 丁香花高清视频完整版在线观看| 18av在线播放| 国产在线小视频| 99综合精品久久| 在线91av| 日本啊v在线| 国产视频二区| 欧美xxxxx性| 中文字幕有码在线视频| av在线免费播放| 国产精品二线| 高清av中文在线字幕观看1| 四虎成人免费| 在线视频福利| 国产亚洲精品一区二区在线观看| 欧美色第一页| gogogo影视剧免费观看在线观看| 国产精品18久久久久网站| 天堂在线看视频| 国产卡1卡2卡三卡在线| 黄网址在线播放免费| а√天堂www在线а√天堂视频 | 日本视频在线观看一区二区三区| 狠狠操视频网| 久色视频在线观看| 亚洲精品aaaa| gogo高清在线播放免费| av丝袜在线| 碰草在线视频| 国产在线中文字幕| 最新av中文字幕| 国产精品视频一区麻豆| 91黄色在线| av中文网站| 黄色三级视频在线观看| 天堂资源最新在线| 中文字幕在线观看av| 国产一区二区三区不卡免费观看| 最新国产在线精品91尤物| 国产网站在线免费观看| 国产导航在线| 日本综合一区二区三区| 高清av中文在线字幕观看1| 国产无遮挡又黄又爽免费网站| 国产黄a三级三级三级av在线看| 国产第一页在线| 青青草免费在线观看| 天堂在线亚洲| 99热国产在线| 丁香在线视频| 99爱视频在线观看| 久艹在线视频| 精品麻豆视频| 国产在线高潮| 怡红院av在线| 国产探花视频在线观看| 波多野结衣中文字幕久久| 2021av天天| 福利视频网站导航| 四虎成人精品在永久免费| 国产一级免费在线观看|