• <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>

            幽魂國度

             

            淺談malloc()和free()

             

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

            1、函數原型及說明:

            void *malloc(long NumBytes)

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

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

            void free(void *FirstByte)

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

            2、函數的用法:

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

            程序代碼:

                     // Code...

                     float *YValue;

                     YValue=(float *)malloc(DataNumberMax*sizeof(float));  //動態分配內存

                     if (NULL == YValue)   exit (1);

                     gets(YValue);

                     // code...

                     free(YValue);

                     YValue= 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));

                 }

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

             

            3free()到底釋放了什么

                free()釋放的是指針指向的內存!注意!釋放的是內存,不是指針!這點非常非常重要!指針是一個變量,只有程序結束時才被銷毀。釋放了內存空間后,原來指向這塊空間的指針還是存在!只不過現在指針指向的內容的垃圾,是未定義的,所以說是垃圾。因此,前面我已經說過了,釋放內存后把指針指向NULL,防止指針在后面不小心又被解引用了。非常重要啊這一點!

             

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

                事實上,仔細看一下free()的函數原型,也許也會發現似乎很神奇,free()函數非常簡單,只有一個參數,只要把指向申請空間的指針傳遞

             

            free()中的參數就可以完成釋放工作!這里要追蹤到malloc()的申請問題了。申請的時候實際上占用的內存要比申請的大。因為超出的空間是用來記錄對這塊內存的管理信息。

            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只能指向可用空間的首地址,不然,減去結構體大小之后一定不是指向管理信息空間的首地址。所以,要確保指針指向可用空間的首地址!
            程序代碼:
            float *YValuePoint;
            while ( !feof(file) && i<DataNumberMax)  //讀數
              {
               fscanf(file,"%f ",&data);      
               YValue[i]=data;
               sum+=YValue[i];
               pLineSerie->AddPoint(XValue++,YValue[i]);
               i++;
              }
              average=sum/DataNumberMax;
              max=*YValue;
              min=*(YValue+2);
             YValuePoint=YValue; //保存首地址
              for(int j=0;j<DataNumberMax;j++)
              {
                  if(max <* YValue)
               {
                    max =* YValue;
               }
               else
               {
                if(min > *YValue)  min = *YValue;
               }
               YValue++;              //YValue地址值+1
              }

              free(YValuePoint);  //釋放內存,釋放從該內存空間的首地址開始

            posted on 2009-12-03 11:50 閱讀(403) 評論(0)  編輯 收藏 引用

            導航

            統計

            常用鏈接

            留言簿

            隨筆檔案

            搜索

            最新評論

            閱讀排行榜

            評論排行榜

            国产 亚洲 欧美 另类 久久| 久久99精品国产麻豆婷婷| 97精品伊人久久久大香线蕉| 四虎国产精品成人免费久久| 草草久久久无码国产专区| 久久99久久99小草精品免视看| 亚洲国产精品高清久久久| 久久久久久精品免费免费自慰| 青青热久久国产久精品| 久久综合视频网站| 欧美与黑人午夜性猛交久久久 | 2021久久精品国产99国产精品| 一本久久知道综合久久| 亚洲AV无码成人网站久久精品大| 久久久黄色大片| 久久国产免费直播| 久久发布国产伦子伦精品| 久久99精品久久久久久| 国产亚洲美女精品久久久| 一日本道伊人久久综合影| 思思久久99热只有频精品66| 伊人久久综合精品无码AV专区| 久久久av波多野一区二区| 亚洲国产精久久久久久久| 久久久久久A亚洲欧洲AV冫| 麻豆精品久久久久久久99蜜桃 | 久久久久人妻一区二区三区| 一本色道久久99一综合| 99999久久久久久亚洲| 久久99精品久久久久久9蜜桃| 久久久久99这里有精品10 | 亚洲中文精品久久久久久不卡| 久久久久久人妻无码| 国产69精品久久久久9999| 99久久香蕉国产线看观香| 成人久久精品一区二区三区| 欧美大战日韩91综合一区婷婷久久青草 | 久久午夜无码鲁丝片午夜精品| 精品国产乱码久久久久久呢| 99久久免费国产精品| 亚洲欧美伊人久久综合一区二区|