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

山寨:不是最好的,是最適合我們的!歡迎體驗(yàn)山寨 中文版MSDN

Blog @ Blog

當(dāng)華美的葉片落盡,生命的脈絡(luò)才歷歷可見。 -- 聶魯達(dá)

常用鏈接

統(tǒng)計(jì)

積分與排名

BBS

Blog

Web

最新評(píng)論

淺談C中的malloc和free

在C語(yǔ)言的學(xué)習(xí)中,對(duì)內(nèi)存管理這部分的知識(shí)掌握尤其重要!之前對(duì)C中的malloc()和free()兩個(gè)函數(shù)的了解甚少,只知道大概該怎么用—— 就是malloc然后free就一切OK了。當(dāng)然現(xiàn)在對(duì)這兩個(gè)函數(shù)的體會(huì)也不見得多,不過對(duì)于本文章第三部分的內(nèi)容倒是有了轉(zhuǎn)折性的認(rèn)識(shí),所以

寫下這篇文章作為一個(gè)對(duì)知識(shí)的總結(jié)。這篇文章之所以命名中有個(gè)“淺談”的字眼,也就是這個(gè)意思了!希望對(duì)大家有一點(diǎn)幫助!

     如果不扯得太遠(yuǎn)的話(比如說(shuō)操作系統(tǒng)中虛擬內(nèi)存和物理內(nèi)存如何運(yùn)做如何管理之類的知識(shí)等),我感覺這篇文章應(yīng)該是比較全面地談了一下malloc()和free().這篇文章由淺入深(不見得有多深)分三個(gè)部分介紹主要內(nèi)容。

廢話了那么多,下面立刻進(jìn)入主題================》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》


一、malloc()和free()的基本概念以及基本用法:

1、函數(shù)原型及說(shuō)明:

void *malloc(long NumBytes):該函數(shù)分配了NumBytes個(gè)字節(jié),并返回了指向這塊內(nèi)存的指針。如果分配失敗,則返回一個(gè)空指針(NULL)。

關(guān)于分配失敗的原因,應(yīng)該有多種,比如說(shuō)空間不足就是一種。

void free(void *FirstByte): 該函數(shù)是將之前用malloc分配的空間還給程序或者是操作系統(tǒng),也就是釋放了這塊內(nèi)存,讓它重新得到自由。

2、函數(shù)的用法:

     其實(shí)這兩個(gè)函數(shù)用起來(lái)倒不是很難,也就是malloc()之后覺得用夠了就甩了它把它給free()了,舉個(gè)簡(jiǎn)單例子:

程序代碼:
        // Code...
        char *Ptr = NULL;
        Ptr = (char *)malloc(100 * sizeof(char));
        if (NULL == Ptr)
    {
        exit (1);
    }
        gets(Ptr);

        // code...
        free(Ptr);
        Ptr = NULL;
        // code...

    就是這樣!當(dāng)然,具體情況要具體分析以及具體解決。比如說(shuō),你定義了一個(gè)指針,在一個(gè)函數(shù)里申請(qǐng)了一塊內(nèi)存然后通過函數(shù)返回傳遞給這個(gè)指針,那么也許釋放這塊內(nèi)存這項(xiàng)工作就應(yīng)該留給其他函數(shù)了。

3、關(guān)于函數(shù)使用需要注意的一些地方:

A、申請(qǐng)了內(nèi)存空間后,必須檢查是否分配成功。

B、當(dāng)不需要再使用申請(qǐng)的內(nèi)存時(shí),記得釋放;釋放后應(yīng)該把指向這塊內(nèi)存的指針指向NULL,防止程序后面不小心使用了它。

C、這兩個(gè)函數(shù)應(yīng)該是配對(duì)。如果申請(qǐng)后不釋放就是內(nèi)存泄露;如果無(wú)故釋放那就是什么也沒有做。釋放只能一次,如果釋放兩次及兩次以上會(huì)

出現(xiàn)錯(cuò)誤(釋放空指針例外,釋放空指針其實(shí)也等于啥也沒做,所以釋放空指針釋放多少次都沒有問題)。

D、雖然malloc()函數(shù)的類型是(void *),任何類型的指針都可以轉(zhuǎn)換成(void *),但是最好還是在前面進(jìn)行強(qiáng)制類型轉(zhuǎn)換,因?yàn)檫@樣可以躲過一

些編譯器的檢查。

好了!最基礎(chǔ)的東西大概這么說(shuō)!現(xiàn)在進(jìn)入第二部分:


二、malloc()到底從哪里得來(lái)了內(nèi)存空間:

1、malloc()到底從哪里得到了內(nèi)存空間?答案是從堆里面獲得空間。也就是說(shuō)函數(shù)返回的指針是指向堆里面的一塊內(nèi)存。操作系統(tǒng)中有一個(gè)記錄空 閑內(nèi)存地址的鏈表。當(dāng)操作系統(tǒng)收到程序的申請(qǐng)時(shí),就會(huì)遍歷該鏈表,然后就尋找第一個(gè)空間大于所申請(qǐng)空間的堆結(jié)點(diǎn),然后就將該結(jié)點(diǎn)從空閑結(jié)點(diǎn)鏈表中刪除,并 將該結(jié)點(diǎn)的空間分配給程序。就是這樣!

   說(shuō)到這里,不得不另外插入一個(gè)小話題,相信大家也知道是什么話題了。什么是堆?說(shuō)到堆,又忍不住說(shuō)到了棧!什么是棧?下面就另外開個(gè)小部分專門而又簡(jiǎn)單地說(shuō)一下這個(gè)題外話:

2、什么是堆:堆是大家共有的空間,分全局堆和局部堆。全局堆就是所有沒有分配的空間,局部堆就是用戶分配的空間。堆在操作系統(tǒng)對(duì)進(jìn)程 初始化的時(shí)候分配,運(yùn)行過程中也可以向系統(tǒng)要額外的堆,但是記得用完了要還給操作系統(tǒng),要不然就是內(nèi)存泄漏。

   什么是棧:棧是線程獨(dú)有的,保存其運(yùn)行狀態(tài)和局部自動(dòng)變量的。棧在線程開始的時(shí)候初始化,每個(gè)線程的棧互相獨(dú)立。每個(gè)函數(shù)都有自己的棧,棧被用來(lái)在函數(shù)之 間傳遞參數(shù)。操作系統(tǒng)在切換線程的時(shí)候會(huì)自動(dòng)的切換棧,就是切換SS/ESP寄存器。棧空間不需要在高級(jí)語(yǔ)言里面顯式的分配和釋放。

   以上的概念描述是標(biāo)準(zhǔn)的描述,不過有個(gè)別語(yǔ)句被我刪除,不知道因?yàn)檫@樣而變得不標(biāo)準(zhǔn)了^_^.

   通過上面對(duì)概念的描述,可以知道:

   棧是由編譯器自動(dòng)分配釋放,存放函數(shù)的參數(shù)值、局部變量的值等。操作方式類似于數(shù)據(jù)結(jié)構(gòu)中的棧。

   堆一般由程序員分配釋放,若不釋放,程序結(jié)束時(shí)可能由OS回收。注意這里說(shuō)是可能,并非一定。所以我想再?gòu)?qiáng)調(diào)一次,記得要釋放!

注意它與數(shù)據(jù)結(jié)構(gòu)中的堆是兩回事,分配方式倒是類似于鏈表。
所以,舉個(gè)例子,如果你在函數(shù)上面定義了一個(gè)指針變量,然后在這個(gè)函數(shù)里申請(qǐng)了一塊內(nèi)存讓指針指向它。實(shí)際上,這個(gè)指針的地址是在棧上,但是它所指向的內(nèi)容卻是在堆上面的!這一點(diǎn)要注意!所以,再想想,在一個(gè)函數(shù)里申請(qǐng)了空間后,比如說(shuō)下面這個(gè)函數(shù):
程序代碼:
   // code...
       void Function(void)
       {
        char *p = (char *)malloc(100 * sizeof(char));
    }


  
   就這個(gè)例子,千萬(wàn)不要認(rèn)為函數(shù)返回,函數(shù)所在的棧被銷毀指針也跟著銷毀,申請(qǐng)的內(nèi)存也就一樣跟著銷毀了!這絕對(duì)是錯(cuò)誤的!因?yàn)樯暾?qǐng)的內(nèi)存在堆上,而函數(shù)所在的棧被銷毀跟堆完全沒有啥關(guān)系。所以,還是那句話:記得釋放!

3、free()到底釋放了什么

   這個(gè)問題比較簡(jiǎn)單,其實(shí)我是想和第二大部分的題目相呼應(yīng)而已!哈哈!free()釋放的是指針指向的內(nèi)存!注意!釋放的是內(nèi)存,不是指針!這點(diǎn)非常非常重 要!指針是一個(gè)變量,只有程序結(jié)束時(shí)才被銷毀。釋放了內(nèi)存空間后,原來(lái)指向這塊空間的指針還是存在!只不過現(xiàn)在指針指向的內(nèi)容的垃圾,是未定義的,所以說(shuō) 是垃圾。因此,前面我已經(jīng)說(shuō)過了,釋放內(nèi)存后把指針指向NULL,防止指針在后面不小心又被解引用了。非常重要啊這一點(diǎn)!

   好了!這個(gè)“題外話”終于說(shuō)完了。就這么簡(jiǎn)單說(shuō)一次,知道個(gè)大概就可以了!下面就進(jìn)入第三個(gè)部分:

三、malloc()以及free()的機(jī)制:

   這個(gè)部分我今天才有了新的認(rèn)識(shí)!而且是轉(zhuǎn)折性的認(rèn)識(shí)!所以,這部分可能會(huì)有更多一些認(rèn)識(shí)上的錯(cuò)誤!不對(duì)的地方請(qǐng)大家?guī)兔χ赋觯?/p>

   事實(shí)上,仔細(xì)看一下free()的函數(shù)原型,也許也會(huì)發(fā)現(xiàn)似乎很神奇,free()函數(shù)非常簡(jiǎn)單,只有一個(gè)參數(shù),只要把指向申請(qǐng)空間的指針傳遞

給free()中的參數(shù)就可以完成釋放工作!這里要追蹤到malloc()的申請(qǐng)問題了。申請(qǐng)的時(shí)候?qū)嶋H上占用的內(nèi)存要比申請(qǐng)的大。因?yàn)槌龅目臻g是用來(lái)記錄對(duì)這塊內(nèi)存的管理信息。先看一下在《UNIX環(huán)境高級(jí)編程》中第七章的一段話:

   大多數(shù)實(shí)現(xiàn)所分配的存儲(chǔ)空間比所要求的要稍大一些,額外的空間用來(lái)記錄管理信息——分配塊的長(zhǎng)度,指向下一個(gè)分配塊的指針等等。這就意味著如果寫過一個(gè)已 分配區(qū)的尾端,則會(huì)改寫后一塊的管理信息。這種類型的錯(cuò)誤是災(zāi)難性的,但是因?yàn)檫@種錯(cuò)誤不會(huì)很快就暴露出來(lái),所以也就很難發(fā)現(xiàn)。將指向分配塊的指針向后移 動(dòng)也可能會(huì)改寫本塊的管理信息。

   以上這段話已經(jīng)給了我們一些信息了。malloc()申請(qǐng)的空間實(shí)際我覺得就是分了兩個(gè)不同性質(zhì)的空間。一個(gè)就是用來(lái)記錄管理信息的空間,另外一個(gè)就是可用空間了。而用來(lái)記錄管理信息的實(shí)際上是一個(gè)結(jié)構(gòu)體。在C語(yǔ)言中,用結(jié)構(gòu)體來(lái)記錄同一個(gè)對(duì)象的不同信息是

天經(jīng)地義的事!下面看看這個(gè)結(jié)構(gòu)體的原型:

程序代碼:
   struct mem_control_block {
    int is_available;    //這是一個(gè)標(biāo)記?
    int size;            //這是實(shí)際空間的大小
    };


  
   對(duì)于size,這個(gè)是實(shí)際空間大小。這里其實(shí)我有個(gè)疑問,is_available是否是一個(gè)標(biāo)記?因?yàn)槲铱戳薴ree()的源代碼之后對(duì)這個(gè)變量感覺有點(diǎn)納悶(源代碼在下面分析)。這里還請(qǐng)大家指出!

   所以,free()就是根據(jù)這個(gè)結(jié)構(gòu)體的信息來(lái)釋放malloc()申請(qǐng)的空間!而結(jié)構(gòu)體的兩個(gè)成員的大小我想應(yīng)該是操作系統(tǒng)的事了。但是這里有一個(gè)問 題,malloc()申請(qǐng)空間后返回一個(gè)指針應(yīng)該是指向第二種空間,也就是可用空間!不然,如果指向管理信息空間的話,寫入的內(nèi)容和結(jié)構(gòu)體的類型有可能不 一致,或者會(huì)把管理信息屏蔽掉,那就沒法釋放內(nèi)存空間了,所以會(huì)發(fā)生錯(cuò)誤!(感覺自己這里說(shuō)的是廢話)

   好了!下面看看free()的源代碼,我自己分析了一下,覺得比起malloc()的源代碼倒是容易簡(jiǎn)單很多。只是有個(gè)疑問,下面指出!

程序代碼:
   // code...
   
       void free(void *ptr) 
    {
            struct mem_control_block *free;
            free = ptr - sizeof(struct mem_control_block);
            free->is_available = 1;
            return;
    }

   看一下函數(shù)第二句,這句非常重要和關(guān)鍵。其實(shí)這句就是把指向可用空間的指針倒回去,讓它指向管理信息的那塊空間,因?yàn)檫@里是在值上減去了一個(gè)結(jié)構(gòu)體的大 小!后面那一句free->is_available = 1;我有點(diǎn)納悶!我的想法是:這里is_available應(yīng)該只是一個(gè)標(biāo)記而已!因?yàn)閺倪@個(gè)變量的名稱上來(lái)看,is_available 翻譯過來(lái)就是“是可以用”。不要說(shuō)我土!我覺得變量名字可以反映一個(gè)變量的作用,特別是嚴(yán)謹(jǐn)?shù)拇a。這是源代碼,所以我覺得絕對(duì)是嚴(yán)謹(jǐn)?shù)模。∵@個(gè)變量的值 是1,表明是可以用的空間!只是這里我想了想,如果把它改為0或者是其他值不知道會(huì)發(fā)生什么事?!但是有一點(diǎn)我可以肯定,就是釋放絕對(duì)不會(huì)那么順利進(jìn)行! 因?yàn)檫@是一個(gè)標(biāo)記!

   當(dāng)然,這里可能還是有人會(huì)有疑問,為什么這樣就可以釋放呢??我剛才也有這個(gè)疑問。后來(lái)我想到,釋放是操作系統(tǒng)的事,那么就free()這個(gè)源代碼來(lái)看, 什么也沒有釋放,對(duì)吧?但是它確實(shí)是確定了管理信息的那塊內(nèi)存的內(nèi)容。所以,free()只是記錄了一些信息,然后告訴操作系統(tǒng)那塊內(nèi)存可以去釋放,具體 怎么告訴操作系統(tǒng)的我不清楚,但我覺得這個(gè)已經(jīng)超出了我這篇文章的討論范圍了。

   那么,我之前有個(gè)錯(cuò)誤的認(rèn)識(shí),就是認(rèn)為指向那塊內(nèi)存的指針不管移到那塊內(nèi)存中的哪個(gè)位置都可以釋放那塊內(nèi)存!但是,這是大錯(cuò)特錯(cuò)!釋放是不可以釋放一部分 的!首先這點(diǎn)應(yīng)該要明白。而且,從free()的源代碼看,ptr只能指向可用空間的首地址,不然,減去結(jié)構(gòu)體大小之后一定不是指向管理信息空間的首地 址。所以,要確保指針指向可用空間的首地址!不信嗎?自己可以寫一個(gè)程序然后移動(dòng)指向可用空間的指針,看程序會(huì)有會(huì)崩!

   最后可能想到malloc()的源代碼看看malloc()到底是怎么分配空間的,這里面涉及到很多其他方面的知識(shí)!有興趣的朋友可以自己去下載源
代碼去看看。


四、關(guān)于其他:

    關(guān)于C中的malloc()和free()的討論就寫到這里吧!寫了三個(gè)鐘頭,感覺有點(diǎn)累!希望對(duì)大家有所幫助!有不對(duì)的地方歡迎大家指出

posted on 2008-03-22 21:49 isabc 閱讀(471) 評(píng)論(1)  編輯 收藏 引用 所屬分類: C++基礎(chǔ)

評(píng)論

# re: 淺談C中的malloc和free 2011-03-11 09:57

不錯(cuò)哈哈,,受教了。。謝謝。  回復(fù)  更多評(píng)論   

廣告信息(免費(fèi)廣告聯(lián)系)

中文版MSDN:
歡迎體驗(yàn)

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            欧美一区二区三区在线看| 久久久不卡网国产精品一区| 美女精品在线| 国产亚洲精品久久久久动| 久久艳片www.17c.com| 亚洲国产91精品在线观看| 欧美精品18| 亚洲丝袜av一区| 亚洲日韩欧美视频| 欧美视频在线观看视频极品| 欧美国产亚洲视频| 国产欧美日韩激情| 亚洲午夜91| 日韩亚洲欧美一区二区三区| 亚洲精品一区二区三| 雨宫琴音一区二区在线| 欧美日韩国产页| 国产情人节一区| 中文亚洲免费| 亚洲一区3d动漫同人无遮挡| 亚洲伊人第一页| 久久在精品线影院精品国产| 蜜臀久久99精品久久久久久9 | 麻豆精品一区二区综合av| 亚洲电影观看| 久久精品夜色噜噜亚洲aⅴ| 玉米视频成人免费看| 久久精品日韩欧美| 欧美大片免费观看在线观看网站推荐| 欧美一区二区在线| 亚洲福利专区| 国内精品久久久久久久影视蜜臀| 免费欧美在线视频| 欧美一区二区视频网站| 久久午夜精品| 欧美aa国产视频| 欧美一区二区免费| 久久亚洲精品欧美| 国产精品99久久99久久久二8| 亚洲区一区二| 欧美一区二区高清| 性伦欧美刺激片在线观看| 欧美在线一二三区| 国产精品大片wwwwww| 国产精品高潮呻吟久久av无限| 亚洲开发第一视频在线播放| 欧美成人国产一区二区| 久久综合给合| 欧美中文字幕视频在线观看| 午夜精品在线| 欧美一区二区三区四区在线 | 欧美激情亚洲自拍| 日韩午夜在线视频| 欧美三区在线| 欧美午夜激情小视频| 亚洲欧洲精品一区| 中文一区二区| 国内精品一区二区三区| 久久一综合视频| 欧美一区永久视频免费观看| 久久综合图片| 亚洲欧洲一区二区在线观看 | 激情综合色综合久久综合| 亚洲在线播放| 午夜免费在线观看精品视频| 国产一在线精品一区在线观看| 亚洲精选视频免费看| 一区二区三区四区五区精品| 亚洲国产精品日韩| 亚洲一区在线视频| 欧美一区三区三区高中清蜜桃| 国产一区在线观看视频| 夜夜嗨av一区二区三区| 99av国产精品欲麻豆| 宅男精品视频| 这里是久久伊人| 亚洲精品乱码视频 | 一区二区日韩伦理片| 欧美日本高清| 亚洲午夜高清视频| 亚洲激情中文1区| 亚洲欧美成人一区二区三区| 亚洲国产欧美在线人成| 欧美黄污视频| 久久国产精品黑丝| 老鸭窝毛片一区二区三区| 日韩视频在线一区二区三区| 欧美美女bbbb| 亚洲欧洲av一区二区| 欧美国产日本在线| 久久蜜臀精品av| 亚洲国产精品毛片| 国产精品网站在线播放| 亚洲欧美日本国产专区一区| 亚洲欧美国产高清| 一本大道久久精品懂色aⅴ | 伊人久久综合97精品| 免费久久99精品国产| 日韩午夜电影av| 亚洲第一中文字幕| 国产午夜精品视频免费不卡69堂| 免费成人在线视频网站| 欧美特黄a级高清免费大片a级| 伊人狠狠色j香婷婷综合| 性欧美精品高清| 欧美在线网站| 久久爱另类一区二区小说| 99在线视频精品| 欧美电影美腿模特1979在线看| 国产亚洲欧美激情| 亚洲福利在线看| 久久久另类综合| 玖玖玖国产精品| 欧美激情第六页| 欧美激情视频在线免费观看 欧美视频免费一 | 一区二区三区欧美视频| 亚洲欧洲综合另类| 亚洲区一区二| 亚洲免费网站| 欧美日韩不卡在线| 亚洲在线观看免费| 国产精品女主播| 影音先锋亚洲一区| 91久久夜色精品国产网站| 亚洲片在线资源| 欧美伊人久久大香线蕉综合69| 亚洲青色在线| 亚洲第一视频网站| 国产综合色一区二区三区| 狠狠色丁香婷婷综合影院| 亚洲福利视频在线| 欧美影院在线| 欧美在线高清视频| 午夜老司机精品| 免费欧美在线| 久久久午夜电影| 亚洲视频第一页| 欧美日韩亚洲高清| 久久国内精品视频| 99亚洲精品| 亚洲欧洲日本在线| 狠狠色综合色区| 国产亚洲在线| 国产精品人人做人人爽人人添| 一区二区欧美国产| 亚洲国产一区二区在线| 欧美fxxxxxx另类| 欧美大片免费观看| 久久琪琪电影院| 欧美激情国产日韩| 亚洲午夜视频在线| 欧美在线999| 欧美日韩在线视频一区| 国产女人水真多18毛片18精品视频 | 欧美激情一区二区| 欧美国产一区二区| 欧美ab在线视频| 国产精品porn| 亚洲精品日韩久久| 美女福利精品视频| 亚洲一区二区三区久久| 久久不见久久见免费视频1| 性色av一区二区怡红| 日韩亚洲欧美一区二区三区| 欧美伊久线香蕉线新在线| 欧美a级理论片| 国产三级欧美三级日产三级99| 一本色道久久综合精品竹菊| 欧美专区日韩视频| 久久久久久999| 欧美99在线视频观看| 亚洲人成小说网站色在线| 久久午夜电影网| 亚洲免费婷婷| 国产欧美亚洲日本| 亚洲资源av| 一本一本久久a久久精品综合妖精 一本一本久久a久久精品综合麻豆 | 午夜精品福利一区二区三区av| 欧美电影专区| 香港久久久电影| 狠狠色丁香婷婷综合影院| 欧美福利精品| 国产伦精品一区二区三区在线观看 | 久久精品人人做人人综合| 久久精品亚洲一区二区三区浴池| 欧美成人一区二区三区| 99热这里只有成人精品国产| 久久激情久久| 午夜欧美精品久久久久久久| 在线播放视频一区| 亚洲精品久久久久久久久| 欧美一区二视频在线免费观看| 国产美女在线精品免费观看| 亚洲高清久久| 亚洲第一色在线| 亚洲高清影视| 亚洲欧洲精品成人久久奇米网| 免费不卡在线观看av| 亚洲最黄网站| 久久精品国产在热久久|