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

life02

  C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理 ::
  197 隨筆 :: 3 文章 :: 37 評論 :: 0 Trackbacks

來自bc-cn C語言論壇

       在C語言的學習中,對內存管理這部分的知識掌握尤其重要!之前對C中的malloc()和free()兩個函數的了解甚少,只知道大概該怎么用——就是malloc然后free就一切OK了。當然現在對這兩個函數的體會也不見得多,不過對于本文章第三部分的內容倒是有了轉折性的認識,所以寫下這篇文章作為一個對知識的總結。這篇文章之所以命名中有個“淺談”的字眼,也就是這個意思了!希望對大家有一點幫助!

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

廢話了那么多,下面立刻進入主題================》》》》》》》》》》》》》》》》》》》》》


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

1、函數原型及說明:

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

關于分配失敗的原因,應該有多種,比如說空間不足就是一種。

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

2、函數的用法:

     其實這兩個函數用起來倒不是很難,也就是malloc()之后覺得用夠了就甩了它把它給free()了,舉個簡單例子:

        // Code...

        char *Ptr = NULL;

        Ptr = (char *)malloc(100 * sizeof(char));

        if (NULL == Ptr){
            exit (1);
        }

        gets(Ptr);

        // code...

        free(Ptr);

        Ptr = NULL;

        // code...

       就是這樣!當然,具體情況要具體分析以及具體解決。比如說,你定義了一個指針,在一個函數里申請了一塊內存然后通過函數返回傳遞給這個指針,那么也許釋放這塊內存這項工作就應該留給其他函數了。

3、關于函數使用需要注意的一些地方:

A、申請了內存空間后,必須檢查是否分配成功。

B、當不需要再使用申請的內存時,記得釋放;釋放后應該把指向這塊內存的指針指向NULL,防止程序后面不小心使用了它。

C、這兩個函數應該是配對。如果申請后不釋放就是內存泄露;如果無故釋放那就是什么也沒有做。釋放只能一次,如果釋放兩次及兩次以上會出現錯誤(釋放空指針例外,釋放空指針其實也等于啥也沒做,所以釋放空指針釋放多少次都沒有問題)。

D、雖然malloc()函數的類型是(void *),任何類型的指針都可以轉換成(void *),但是最好還是在前面進行強制類型轉換,因為這樣可以躲過一些編譯器的檢查。


二、malloc()到底從哪里得來了內存空間:

1、malloc()到底從哪里得到了內存空間?答案是從堆里面獲得空間。也就是說函數返回的指針是指向堆里面的一塊內存。操作系統中有一個記錄空閑內存地址的鏈表。當操作系統收到程序的申請時,就會遍歷該鏈表,然后就尋找第一個空間大于所申請空間的堆結點,然后就將該結點從空閑結點鏈表中刪除,并將該結點的空間分配給程序。就是這樣!

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

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

   什么是棧:棧是線程獨有的,保存其運行狀態和局部自動變量的。棧在線程開始的時候初始化,每個線程的棧互相獨立。每個函數都有自己的棧,棧被用來在函數之間傳遞參數。操作系統在切換線程的時候會自動的切換棧,就是切換SS/ESP寄存器。棧空間不需要在高級語言里面顯式的分配和釋放。

   以上的概念描述是標準的描述,不過有個別語句被我刪除,不知道因為這樣而變得不標準了^_^.

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

   棧是由編譯器自動分配釋放,存放函數的參數值、局部變量的值等。操作方式類似于數據結構中的棧。

   堆一般由程序員分配釋放,若不釋放,程序結束時可能由OS回收。注意這里說是可能,并非一定。所以我想再強調一次,記得要釋放!

注意它與數據結構中的堆是兩回事,分配方式倒是類似于鏈表。(這點我上面稍微提過)


   所以,舉個例子,如果你在函數上面定義了一個指針變量,然后在這個函數里申請了一塊內存讓指針指向它。實際上,這個指針的地址是在棧上,但是它所指向的內容卻是在堆上面的!這一點要注意!所以,再想想,在一個函數里申請了空間后,比如說下面這個函數:

   // code...

       void Function(void)
       {
           char *p = (char *)malloc(100 * sizeof(char));
       }

       就這個例子,千萬不要認為函數返回,函數所在的棧被銷毀指針也跟著銷毀,申請的內存也就一樣跟著銷毀了!這絕對是錯誤的!因為申請的內存在堆上,而函數所在的棧被銷毀跟堆完全沒有啥關系。所以,還是那句話:記得釋放!

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

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

       好了!這個“題外話”終于說完了。就這么簡單說一次,知道個大概就可以了!下面就進入第三個部分:

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

       這個部分我今天才有了新的認識!而且是轉折性的認識!所以,這部分可能會有更多一些認識上的錯誤!不對的地方請大家幫忙指出!

       事實上,仔細看一下free()的函數原型,也許也會發現似乎很神奇,free()函數非常簡單,只有一個參數,只要把指向申請空間的指針傳遞給free()中的參數就可以完成釋放工作!這里要追蹤到malloc()的申請問題了。申請的時候實際上占用的內存要比申請的大。因為超出的空間是用來記錄對這塊內存的管理信息。先看一下在《UNIX環境高級編程》中第七章的一段話:

       大多數實現所分配的存儲空間比所要求的要稍大一些,額外的空間用來記錄管理信息——分配塊的長度,指向下一個分配塊的指針等等。這就意味著如果寫過一個已分配區的尾端,則會改寫后一塊的管理信息。這種類型的錯誤是災難性的,但是因為這種錯誤不會很快就暴露出來,所以也就很難發現。將指向分配塊的指針向后移動也可能會改寫本塊的管理信息。

       以上這段話已經給了我們一些信息了。malloc()申請的空間實際我覺得就是分了兩個不同性質的空間。一個就是用來記錄管理信息的空間,另外一個就是可用空間了。而用來記錄管理信息的實際上是一個結構體。在C語言中,用結構體來記錄同一個對象的不同信息是天經地義的事!下面看看這個結構體的原型:

   struct mem_control_block {

       int is_available;    //這是一個標記?

       int size;            //這是實際空間的大小

    };

       對于size,這個是實際空間大小。這里其實我有個疑問,is_available是否是一個標記?因為我看了free()的源代碼之后對這個變量感覺有點納悶(源代碼在下面分析)。這里還請大家指出!

       所以,free()就是根據這個結構體的信息來釋放malloc()申請的空間!而結構體的兩個成員的大小我想應該是操作系統的事了。但是這里有一個問題,malloc()申請空間后返回一個指針應該是指向第二種空間,也就是可用空間!不然,如果指向管理信息空間的話,寫入的內容和結構體的類型有可能不一致,或者會把管理信息屏蔽掉,那就沒法釋放內存空間了,所以會發生錯誤!(感覺自己這里說的是廢話)

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

   // code...

  
       void free(void *ptr)
       {

            struct mem_control_block *free;

            free = ptr - sizeof(struct mem_control_block);

            free->is_available = 1;

            return;
       }

        看一下函數第二句,這句非常重要和關鍵。其實這句就是把指向可用空間的指針倒回去,讓它指向管理信息的那塊空間,因為這里是在值上減去了一個結構體的大小!后面那一句free->is_available = 1;我有點納悶!我的想法是:這里is_available應該只是一個標記而已!因為從這個變量的名稱上來看,is_available 翻譯過來就是“是可以用”。不要說我土!我覺得變量名字可以反映一個變量的作用,特別是嚴謹的代碼。這是源代碼,所以我覺得絕對是嚴謹的!!這個變量的值是1,表明是可以用的空間!只是這里我想了想,如果把它改為0或者是其他值不知道會發生什么事?!但是有一點我可以肯定,就是釋放絕對不會那么順利進行!因為這是一個標記!

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

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

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


四、關于其他:

        關于C中的malloc()和free()的討論就寫到這里吧!寫了三個鐘頭,感覺有點累!希望對大家有所幫助!有不對的地方歡迎大家指出!最后,謝謝參與這個帖子討論的所有朋友,帖子:http://www.bc-cn.net/bbs/dispbbs.asp?boardID=5&ID=81781&page=1

posted on 2009-10-09 13:02 life02 閱讀(247) 評論(0)  編輯 收藏 引用 所屬分類: c++學習
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            一区二区在线视频播放| 欧美国产日韩免费| 在线日韩av片| 国产一区二区中文字幕免费看| 欧美日韩在线播放三区| 欧美日韩国产二区| 欧美日韩国产不卡在线看| 欧美乱妇高清无乱码| 欧美日韩高清在线| 国产精品大片免费观看| 国产日韩精品一区二区| 在线视频成人| 在线视频欧美精品| 欧美亚洲免费高清在线观看| 久久综合成人精品亚洲另类欧美| 嫩草伊人久久精品少妇av杨幂| 欧美黑人国产人伦爽爽爽| 亚洲精品一区二区三区婷婷月| 99精品欧美一区二区三区| 欧美在线看片a免费观看| 亚洲欧美一区二区在线观看| 欧美一区二区三区在线播放| 久久综合精品一区| av成人动漫| 久久婷婷蜜乳一本欲蜜臀| 欧美日韩精品不卡| 极品尤物一区二区三区| 亚洲视频日本| 能在线观看的日韩av| 日韩午夜三级在线| 久久婷婷国产综合精品青草| 欧美成人国产va精品日本一级| 国产欧美精品一区二区色综合| 亚洲成人资源| 亚洲欧美国产77777| 欧美电影美腿模特1979在线看| 国产精品99久久久久久久vr| 久久一区二区三区国产精品| 国产精品你懂的在线欣赏| 亚洲区一区二区三区| 久久久国产一区二区| 日韩一级在线| 欧美激情91| 亚洲国产高潮在线观看| 久久久久久成人| 一区二区三区www| 欧美精品一区二区三区一线天视频| 国产一区二区三区免费在线观看 | 亚洲国产精品久久91精品| 一区二区电影免费在线观看| 欧美本精品男人aⅴ天堂| 亚洲欧美日韩精品久久| 欧美日韩网站| 一区二区三区四区在线| 亚洲国产免费| 亚洲综合社区| 欧美三级日韩三级国产三级 | 亚洲欧美影院| 999亚洲国产精| 欧美大片在线看| 91久久久久久久久| 免费成人在线视频网站| 久久国产精品亚洲77777| 国产精品无人区| 先锋影院在线亚洲| 午夜一区二区三区在线观看| 国产乱码精品一区二区三区五月婷| 亚洲小说春色综合另类电影| 99re成人精品视频| 国产精品成人一区| 欧美一级久久久| 亚洲欧美在线高清| 国产中文一区二区| 欧美sm极限捆绑bd| 欧美精品久久99| 中文久久精品| 亚洲色诱最新| 国产欧美日韩亚洲精品| 欧美在线播放| 日韩小视频在线观看专区| 欧美成人亚洲成人| 亚洲视频精品| 亚欧成人在线| 亚洲国产第一页| 99精品欧美一区二区蜜桃免费| 国产精品国产三级国产| 久久精品成人| 免费不卡视频| 亚洲综合视频一区| 久久久成人网| 亚洲视频欧美在线| 久久精品青青大伊人av| 日韩午夜三级在线| 午夜视频久久久| 亚洲级视频在线观看免费1级| 99热在这里有精品免费| 国内在线观看一区二区三区| 亚洲人成在线观看一区二区| 欧美日韩一区二区三区在线观看免| 久久成人综合视频| 欧美成人福利视频| 久久激情视频| 欧美激情第六页| 久久久久久69| 国产精品va在线播放| 麻豆精品视频| 国产精品爱久久久久久久| 鲁大师影院一区二区三区| 欧美日韩亚洲免费| 欧美成人精品激情在线观看| 亚洲性xxxx| 亚洲欧美综合网| 老司机午夜精品视频在线观看| 亚洲欧美国产77777| 麻豆精品视频在线观看| 欧美一区二区国产| 欧美理论电影在线观看| 另类av一区二区| 国产精品久久久亚洲一区| 美女爽到呻吟久久久久| 欧美色图一区二区三区| 亚洲国产三级网| 尤物九九久久国产精品的特点| 亚洲一区国产精品| 亚洲性色视频| 欧美日韩国产一区二区| 亚洲高清在线观看| 激情成人亚洲| 欧美在线观看视频| 久久国产婷婷国产香蕉| 国产欧美日韩综合一区在线播放| 亚洲免费观看高清完整版在线观看熊 | 模特精品裸拍一区| 久久久水蜜桃av免费网站| 国产精品日产欧美久久久久| 亚洲精品社区| 日韩视频―中文字幕| 欧美大片第1页| 亚洲电影欧美电影有声小说| 久久综合给合久久狠狠色| 一区二区三区精品视频在线观看| 亚洲精品偷拍| 欧美精品色网| 最新日韩av| 一区二区精品国产| 欧美日韩中文另类| 亚洲一区在线观看免费观看电影高清| 中文网丁香综合网| 国产精品www.| 午夜精品区一区二区三| 久久蜜桃精品| 亚洲国产欧美在线人成| 欧美福利网址| 一区二区三区 在线观看视| 午夜亚洲激情| 国模私拍视频一区| 玖玖玖国产精品| 亚洲美女免费精品视频在线观看| 中文日韩在线视频| 国产在线成人| 欧美高清在线一区| 亚洲一级影院| 久久综合久色欧美综合狠狠| 亚洲日本一区二区| 国产精品vvv| 久久久久**毛片大全| 亚洲精品久久久蜜桃| 欧美亚洲色图校园春色| 在线看成人片| 国产精品大片wwwwww| 久久久久久久久蜜桃| 亚洲精品黄色| 久久国产精品网站| 日韩视频一区二区三区在线播放| 国产精品狼人久久影院观看方式| 久久久91精品| 欧美国产视频在线观看| 欧美一区二区三区视频在线观看| 国产精品a久久久久久| 久久人人超碰| 国产精品99久久久久久www| 欧美成人精品在线视频| 亚洲欧美卡通另类91av | 亚洲精品久久视频| 国产九九精品视频| 欧美伦理a级免费电影| 欧美一二三区精品| 亚洲免费激情| 欧美成人网在线| 久久本道综合色狠狠五月| 亚洲精品在线免费| 国产一区二区精品久久| 欧美精品在线一区| 久久天堂av综合合色| 亚洲欧美国产日韩中文字幕| 亚洲欧洲在线一区| 美女久久网站| 欧美在线三区| 午夜免费电影一区在线观看| 亚洲人成网站999久久久综合 |