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

            內(nèi)存泄漏


            本文來自http://blog.csdn.net/zxcred

             C++程序的復(fù)雜性很大一部分在于他的內(nèi)存管理,沒有C#那樣的垃圾回收機(jī)制,內(nèi)存管理對(duì)初學(xué)者來說很困難。經(jīng)常會(huì)出現(xiàn)內(nèi)存泄露的情況。那么我們寫程序如何避免內(nèi)存泄露呢?首先我們需要知道程序有沒有內(nèi)存泄露,然后定位到底是哪行代碼出現(xiàn)內(nèi)存泄露了,這樣才能將其修復(fù)。
             本文描述了如何檢測(cè)內(nèi)存泄露。最主要的是純C,C++的程序如何檢測(cè)內(nèi)存泄露。
             現(xiàn)在有很多專業(yè)的檢測(cè)工具,比如比較有名的BoundsCheck, 但是這類工具也有他的缺點(diǎn),我認(rèn)為首先BoundsCheck是商業(yè)軟件,呵呵。然后呢需要安裝,使用起來不太方便。因?yàn)槲覀儥z測(cè)的時(shí)候不一定經(jīng)常會(huì)啟動(dòng)他來檢測(cè)。這樣經(jīng)常會(huì)積累很多問題,那時(shí)要解決就麻煩了。最好就是從開始編碼,一步一步的都能隨時(shí)提醒我們內(nèi)存泄露。我們編程序會(huì)經(jīng)常調(diào)試,假如能在每次調(diào)試程序的時(shí)候都能自動(dòng)檢測(cè)內(nèi)存泄露就好了。

              一. 在 MFC 中檢測(cè)內(nèi)存泄漏
            假如是用MFC的程序的話,很簡(jiǎn)單。默認(rèn)的就有內(nèi)存泄露檢測(cè)的功能。
            我們用VS2005生成了一個(gè)MFC的對(duì)話框的程序,發(fā)現(xiàn)他可以自動(dòng)的檢測(cè)內(nèi)存泄露.不用我們做任何特殊的操作. 仔細(xì)觀察,發(fā)現(xiàn)在每個(gè)CPP文件中,都有下面的代碼:

            1. #ifdef _DEBUG
            2. #define new DEBUG_NEW
            3. #endif
            DEBUG_NEW 這個(gè)宏定義在afx.h文件中,就是它幫助我們定位內(nèi)存泄漏。

                在含有以上代碼的cpp文件中分配內(nèi)存后假如沒有刪除,那么停止程序的時(shí)候,VisualStudio的Output窗口就會(huì)顯示如下的信息了:
            1. Detected memory leaks!
            2. Dumping objects ->
            3. d:\code\mfctest\mfctest.cpp(80) : {157normal block at 0x003AF1704 bytes long.
            4.  Data: < > 00 00 00 00 
            5. Object dump complete.

                在Output窗口雙擊粗體字那一行,那么IDE就會(huì)打開該文件,定位到該行,很容易看出是哪出現(xiàn)了內(nèi)存泄露。

                二.檢測(cè)純C++的程序內(nèi)存泄露

            我試了下用VisualStudio建立的Win32 Console Application和Win32 Project項(xiàng)目,結(jié)果都不能檢測(cè)出內(nèi)存泄露。
            下面一步一步來把程序的內(nèi)存泄露檢測(cè)的機(jī)制建立起來。
            首先,我們需要知道C運(yùn)行庫的Debug版本提供了許多檢測(cè)功能,使得我們更容易的Debug程序。在MSDN中有專門的章節(jié)講這個(gè),叫做Debug Routines,建議大家先看看里面的內(nèi)容吧。
            我們會(huì)用到里面很重要的幾個(gè)函數(shù)。其中最重要的是 _CrtDumpMemoryLeaks();自己看MSDN里的幫助吧。使用這個(gè)函數(shù),需要包含頭文件crtdbg.h
            該函數(shù)只在Debug版本才有用,當(dāng)在調(diào)試器下運(yùn)行程序時(shí),_CrtDumpMemoryLeaks 將在“Output(輸出)”窗口中顯示內(nèi)存泄漏信息.寫段代碼試驗(yàn)一下吧,如下:
              檢測(cè)內(nèi)存泄露版本一
            1. #include "stdafx.h"
            2. #include <crtdbg.h>
            3. int _tmain(int argc, _TCHAR* argv[])
            4. {
            5.     int* p = new int();
            6.     _CrtDumpMemoryLeaks();
            7.     return 0;
            8. }

              運(yùn)行后,在Output(輸出)窗口,顯示了如下的信息:

            1. Detected memory leaks!
            2. Dumping objects ->
            3. {112} normal block at 0x003AA770, 4 bytes long.
            4.  Data: <    > 00 00 00 00 
            5. Object dump complete.

              但是這個(gè)只是告訴我們程序有內(nèi)存泄露,到底在哪泄露了一眼看不出來啊。
              看我們的檢測(cè)內(nèi)存泄露版本二

            1. #include "stdafx.h"
            2. #ifdef _DEBUG
            3. #define DEBUG_CLIENTBLOCK   new( _CLIENT_BLOCK, __FILE__, __LINE__)
            4. #else
            5. #define DEBUG_CLIENTBLOCK
            6. #endif
            7. #define _CRTDBG_MAP_ALLOC
            8. #include <crtdbg.h>
            9. #ifdef _DEBUG
            10. #define new DEBUG_CLIENTBLOCK
            11. #endif
            12. int _tmain(int argc, _TCHAR* argv[])
            13. {
            14.     int* p = new int();
            15.     _CrtDumpMemoryLeaks();
            16.     return 0;
            17. }
              該程序定義了幾個(gè)宏,通過宏將Debug版本下的new給替換了,新的new記錄下了調(diào)用new時(shí)的文件名和代碼行.運(yùn)行后,可以看到如下的結(jié)果:

            1. Detected memory leaks!
            2. Dumping objects ->
            3. d:\code\consoletest\consoletest.cpp(21) : {112} client block at 0x003A38B0, subtype 0, 4 bytes long.
            4.  Data: <    > 00 00 00 00 
            5. Object dump complete.
              呵呵,已經(jīng)和MFC程序的效果一樣了,但是等一等。看下如下的代碼吧:
            1. int _tmain(int argc, _TCHAR* argv[])
            2. {
            3.     int* p = new int();
            4.     _CrtDumpMemoryLeaks();
            5.     delete p;
            6.     return 0;
            7. }

              運(yùn)行后可以發(fā)現(xiàn)我們刪除了指針,但是它仍然報(bào)內(nèi)存泄露。所以可以想象,每調(diào)用一次new,程序內(nèi)部都會(huì)將該調(diào)用記錄下來,類似于有個(gè)數(shù)組記錄,假如delete了,那么就將其從數(shù)組中刪除,而_CrtDumpMemoryLeaks()就是把這個(gè)數(shù)組當(dāng)前的狀態(tài)打印出來。
            所以除了在必要的時(shí)候Dump出內(nèi)存信息外,最重要的就是在程序退出的時(shí)候需要掉用一次_CrtDumpMemoryLeaks();
             假如程序有不止一個(gè)出口,那么我們就需要在多個(gè)地方都調(diào)用該函數(shù)。
             更進(jìn)一步,假如程序在類的析構(gòu)函數(shù)里刪除指針,怎么辦?例如:

            1. #include "stdafx.h"
            2. #ifdef _DEBUG
            3. #define DEBUG_CLIENTBLOCK   new( _CLIENT_BLOCK, __FILE__, __LINE__)
            4. #else
            5. #define DEBUG_CLIENTBLOCK
            6. #endif
            7. #define _CRTDBG_MAP_ALLOC
            8. #include <crtdbg.h>
            9. #ifdef _DEBUG
            10. #define new DEBUG_CLIENTBLOCK
            11. #endif
            12. class Test
            13. {
            14. public:
            15.     Test()      {   _p = new int();     }
            16.     ~Test()     {   delete _p;          }
            17.     int* _p;
            18. };
            19. int _tmain(int argc, _TCHAR* argv[])
            20. {
            21.     int* p = new int();
            22.     delete p;
            23.     Test t;
            24.     _CrtDumpMemoryLeaks();
            25.     return 0;
            26. }
              可以看到析構(gòu)函數(shù)在程序退出的時(shí)候才調(diào)用,明明沒有內(nèi)存泄露,但是這樣的寫法還是報(bào)了。
              如何改進(jìn)呢,看檢測(cè)內(nèi)存泄露版本三
            1. #include "stdafx.h"
            2. #ifdef _DEBUG
            3. #define DEBUG_CLIENTBLOCK   new( _CLIENT_BLOCK, __FILE__, __LINE__)
            4. #else
            5. #define DEBUG_CLIENTBLOCK
            6. #endif
            7. #define _CRTDBG_MAP_ALLOC
            8. #include <crtdbg.h>
            9. #ifdef _DEBUG
            10. #define new DEBUG_CLIENTBLOCK
            11. #endif
            12. class Test
            13. {
            14. public:
            15.     Test()      {   _p = new int();     }
            16.     ~Test()     {   delete _p;          }
            17.     int* _p;
            18. };
            19. int _tmain(int argc, _TCHAR* argv[])
            20. {
            21.     _CrtSetDbgFlag ( _CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF );
            22.     int* p = new int();
            23.     delete p;
            24.     Test t;
            25.     return 0;
            26. }

              _CrtSetDbgFlag ( _CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF );
            該語句在程序退出時(shí)自動(dòng)調(diào)用 _CrtDumpMemoryLeaks。必須同時(shí)設(shè)置 _CRTDBG_ALLOC_MEM_DF 和 _CRTDBG_LEAK_CHECK_DF.
              這樣,該版本已經(jīng)達(dá)到了MFC一樣的效果了,但是我覺得光這樣還不夠,因?yàn)槲覀冎皇窃贠utput窗口中輸出信息,對(duì)開發(fā)人員的提醒還不明顯,經(jīng)常會(huì)被遺漏,而且很多人就算發(fā)現(xiàn)了內(nèi)存泄露,但是不好修復(fù),不會(huì)嚴(yán)重影響到程序外在表現(xiàn),都不會(huì)修復(fù)。怎么樣能讓開發(fā)人員主動(dòng)的修復(fù)內(nèi)存泄露的問題呢?記得曾經(jīng)和人配合寫程序,我的函數(shù)參數(shù)有要求,不能為空,但是別人老是傳空值,沒辦法了,只好在函數(shù)開始驗(yàn)證函數(shù)參數(shù),給他assert住,這樣程序運(yùn)行時(shí)老是不停的彈出assert,調(diào)試程序那個(gè)煩壓,最后其他程序員煩了,就把這個(gè)問題給改好了,輸入?yún)?shù)就正確了。所以我覺得咱要讓程序員主動(dòng)去做一件事,首先要讓他覺得做這個(gè)事是能減輕自己負(fù)擔(dān),讓自己工作輕松的。呵呵,那咱們也這樣,當(dāng)程序退出時(shí),檢測(cè)到內(nèi)存泄露就讓程序提示出來。
              看檢測(cè)內(nèi)存泄露版本四

            1. #include "stdafx.h"
            2. #include <assert.h>
            3. #ifdef _DEBUG
            4. #define DEBUG_CLIENTBLOCK   new( _CLIENT_BLOCK, __FILE__, __LINE__)
            5. #else
            6. #define DEBUG_CLIENTBLOCK
            7. #endif
            8. #define _CRTDBG_MAP_ALLOC
            9. #include <crtdbg.h>
            10. #ifdef _DEBUG
            11. #define new DEBUG_CLIENTBLOCK
            12. #endif
            13. void Exit()
            14. {
            15.     int i = _CrtDumpMemoryLeaks();
            16.     assert( i == 0);
            17. }
            18. int _tmain(int argc, _TCHAR* argv[])
            19. {
            20.     atexit(Exit);
            21.     int* p = new int();
            22.     return 0;
            23. }
              該版本會(huì)在程序退出時(shí)檢查內(nèi)存泄露,假如存在就會(huì)彈出提示對(duì)話框.
             atexit(Exit);設(shè)置了在程序退出時(shí)執(zhí)行Exit()函數(shù)。
             Exit()函數(shù)中,假如存在內(nèi)存泄露,_CrtDumpMemoryLeaks()會(huì)返回非0值,就會(huì)被assert住了。

               到這個(gè)版本已經(jīng)達(dá)到可以使用的程度了。但是我們還可以做些改進(jìn),因?yàn)檎嬉獪?zhǔn)確的檢測(cè)到代碼中所有的內(nèi)存泄露,需要把代碼中的#define……拷貝到所有使用new的文件中。不可能每個(gè)文件都拷貝這么多代碼,所以我們可以將他提取出來,放在一個(gè)文件中,比如我是放在KDetectMemoryLeak.h中,該文件內(nèi)容如下:

            1. #pragma once
            2. #ifdef _DEBUG
            3. #define DEBUG_CLIENTBLOCK   new( _CLIENT_BLOCK, __FILE__, __LINE__)
            4. #else
            5. #define DEBUG_CLIENTBLOCK
            6. #endif
            7. #define _CRTDBG_MAP_ALLOC
            8. #include <stdlib.h>
            9. #include <crtdbg.h>
            10. #ifdef _DEBUG
            11. #define new DEBUG_CLIENTBLOCK
            12. #endif
              然后將KDetectMemoryLeak.h包含在項(xiàng)目的通用文件中,例如用VS建的項(xiàng)目就將其包含在stdafx.h中。或者我自己建的一個(gè)Common.h文件中,該文件包含一些通用的,基本所有文件都會(huì)用到的代碼東東。

              好了,到現(xiàn)在,檢測(cè)內(nèi)存泄露總算完成了,而且他還能定位到到底是代碼中哪個(gè)文件,哪行出現(xiàn)了內(nèi)存泄露。下一篇文章將會(huì)講些實(shí)際遇到的一些問題,例如只知道有內(nèi)存泄露,但是不知道到底內(nèi)存泄露的具體位置,如何利用內(nèi)存斷點(diǎn)等技術(shù)來定位內(nèi)存泄露的位置啊,最后會(huì)從代碼的角度講下,怎么樣才能避免內(nèi)存泄露吧。

            posted on 2009-06-03 12:57 Brandon 閱讀(203) 評(píng)論(0)  編輯 收藏 引用


            只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


            <2025年6月>
            25262728293031
            1234567
            891011121314
            15161718192021
            22232425262728
            293012345

            導(dǎo)航

            統(tǒng)計(jì)

            常用鏈接

            留言簿(2)

            隨筆分類

            隨筆檔案

            文章分類

            文章檔案

            收藏夾

            IT WEB

            常用鏈接

            牛人BLOG

            學(xué)習(xí)網(wǎng)站

            搜索

            積分與排名

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

            久久国产精品无码一区二区三区 | 亚洲国产成人久久综合碰碰动漫3d| 精品久久国产一区二区三区香蕉 | 久久久无码精品午夜| 99999久久久久久亚洲| 久久精品成人欧美大片| 亚洲国产精品嫩草影院久久| 精品久久久久久国产牛牛app| 26uuu久久五月天| 国产精品99久久不卡| 久久99精品国产99久久6| 久久精品国产99国产精品| 精品久久久久久国产三级| 久久久久亚洲AV无码去区首| 理论片午午伦夜理片久久 | 久久AV无码精品人妻糸列| 亚洲欧美一区二区三区久久| 久久国产成人精品国产成人亚洲| 97久久久精品综合88久久| 精品永久久福利一区二区| avtt天堂网久久精品| 亚洲欧美日韩精品久久| 国产精品久久久99| 无码乱码观看精品久久| 精品久久久久久无码专区 | 国产精品成人久久久久久久 | 日韩十八禁一区二区久久| 无码国内精品久久人妻麻豆按摩| 亚洲欧美一级久久精品| 久久人人添人人爽添人人片牛牛| 色婷婷综合久久久久中文一区二区 | 狠狠色狠狠色综合久久| 久久婷婷五月综合97色| 人人狠狠综合久久亚洲88| 久久久99精品一区二区| 人妻无码αv中文字幕久久琪琪布| 人人狠狠综合久久亚洲88| 欧美大战日韩91综合一区婷婷久久青草 | 亚洲AV无码久久| 久久精品成人免费网站| 要久久爱在线免费观看|