日本综合一区二区|亚洲中文天堂综合|日韩欧美自拍一区|男女精品天堂一区|欧美自拍第6页亚洲成人精品一区|亚洲黄色天堂一区二区成人|超碰91偷拍第一页|日韩av夜夜嗨中文字幕|久久蜜综合视频官网|精美人妻一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時(shí)間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營(yíng)銷解決方案
檢測(cè)C++中的內(nèi)存泄漏

對(duì)于C/C++程序員來說,內(nèi)存泄漏是一個(gè)常見的也是令人頭疼的問題。一般我們常說的內(nèi)存泄漏是指對(duì)內(nèi)存的泄漏對(duì)內(nèi)存是指程序從堆中分配的,大小任意的,使用完后必須顯示釋放的內(nèi)存。

成都創(chuàng)新互聯(lián)服務(wù)項(xiàng)目包括達(dá)茂旗網(wǎng)站建設(shè)、達(dá)茂旗網(wǎng)站制作、達(dá)茂旗網(wǎng)頁制作以及達(dá)茂旗網(wǎng)絡(luò)營(yíng)銷策劃等。多年來,我們專注于互聯(lián)網(wǎng)行業(yè),利用自身積累的技術(shù)優(yōu)勢(shì)、行業(yè)經(jīng)驗(yàn)、深度合作伙伴關(guān)系等,向廣大中小型企業(yè)、政府機(jī)構(gòu)等提供互聯(lián)網(wǎng)行業(yè)的解決方案,達(dá)茂旗網(wǎng)站推廣取得了明顯的社會(huì)效益與經(jīng)濟(jì)效益。目前,我們服務(wù)的客戶以成都為中心已經(jīng)輻射到達(dá)茂旗省份的部分城市,未來相信會(huì)繼續(xù)擴(kuò)大服務(wù)區(qū)域并繼續(xù)獲得客戶的支持與信任!

首先我們需要知道程序有沒有內(nèi)存泄露,然后定位到底是哪行代碼出現(xiàn)內(nèi)存泄露了,這樣才能將其修復(fù)。

最簡(jiǎn)單的方法當(dāng)然是借助于專業(yè)的檢測(cè)工具,比較有名如BoundsCheck,功能非常強(qiáng)大,相信做C++開發(fā)的人都離不開它。此外就是不使用任何工具,而是自己來實(shí)現(xiàn)對(duì)內(nèi)存泄露的監(jiān)控,分如下兩種情況:

一. 在 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) : {157} normal block at 0x003AF170, 4 bytes long.  
  4. Data: < > 00 00 00 00  
  5. Object dump complete. 

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

#p#

二.檢測(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   
  3. int _tmain(int argc, _TCHAR* argv[])  
  4. {  
  5. int* p = new int;  
  6. _CrtDumpMemoryLeaks;  
  7. return 0;  

運(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   
  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;  

該程序定義了幾個(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程序的效果一樣了,但是等一等??聪氯缦碌拇a吧:

 
 
 
  1. int _tmain(int argc, _TCHAR* argv[])  
  2. {  
  3. int* p = new int;  
  4. _CrtDumpMemoryLeaks;  
  5. delete p;  
  6. return 0;  

運(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   
  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;  

可以看到析構(gòu)函數(shù)在程序退出的時(shí)候才調(diào)用,明明沒有內(nèi)存泄露,但是這樣的寫法還是報(bào)了。

#p#

如何改進(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   
  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;  
 
 
 
  1. _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   
  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   
  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;  

該版本會(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   
  9. #include   
  10. #ifdef _DEBUG  
  11. #define new DEBUG_CLIENTBLOCK  
  12. #endif 

然后將KDetectMemoryLeak.h包含在項(xiàng)目的通用文件中,例如用VS建的項(xiàng)目就將其包含在stdafx.h中。或者我自己建的一個(gè)Common.h文件中,該文件包含一些通用的,基本所有文件都會(huì)用到的代碼。

【編輯推薦】

  1. 在C/C++算法設(shè)計(jì)中使用任意位寬
  2. C++程序中可以命名的5種元素
  3. VC++獲得當(dāng)前系統(tǒng)時(shí)間的幾種方案
  4. 影響C++/C程序的幾大要素
  5. C++連接mysql數(shù)據(jù)庫的兩種方法

網(wǎng)站題目:檢測(cè)C++中的內(nèi)存泄漏
網(wǎng)頁網(wǎng)址:http://www.dlmjj.cn/article/cojihio.html