欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C++中關(guān)于Crt的內(nèi)存泄漏檢測(cè)的分析介紹

 更新時(shí)間:2013年04月28日 12:40:49   作者:  
本篇文章介紹了,在C++中關(guān)于Crt的內(nèi)存泄漏檢測(cè)的分析說(shuō)明。需要的朋友參考下

盡管這個(gè)概念已經(jīng)讓人說(shuō)濫了 ,還是想簡(jiǎn)單記錄一下, 以備以后查詢(xún)。

復(fù)制代碼 代碼如下:

#ifdef _DEBUG
#define DEBUG_CLIENTBLOCK   new( _CLIENT_BLOCK, __FILE__, __LINE__)
#else
#define DEBUG_CLIENTBLOCK
#endif
#define _CRTDBG_MAP_ALLOC
#include <crtdbg.h>
#ifdef _DEBUG
#define new DEBUG_CLIENTBLOCK
#endif


int _tmain(int argc, _TCHAR* argv[])
{
    char* p = new char();
    char* pp = new char[10];
    char* ppp = (char*)malloc(10);

    _CrtDumpMemoryLeaks();

    return 0;
}


主要原理是運(yùn)用Crt 的內(nèi)存調(diào)試功能, 通過(guò)宏替代默認(rèn)的operator new, 讓它被下面版本替代:
復(fù)制代碼 代碼如下:

void *__CRTDECL operator new(
        size_t cb,
        int nBlockUse,
        const char * szFileName,
        int nLine
        )
        _THROW1(_STD bad_alloc)
{
    /* _nh_malloc_dbg already calls _heap_alloc_dbg in a loop and calls _callnewh
       if the allocation fails. If _callnewh returns (very likely because no
       new handlers have been installed by the user), _nh_malloc_dbg returns NULL.
     */
    void *res = _nh_malloc_dbg( cb, 1, nBlockUse, szFileName, nLine );

    RTCCALLBACK(_RTC_Allocate_hook, (res, cb, 0));

    /* if the allocation fails, we throw std::bad_alloc */
    if (res == 0)
    {
        static const std::bad_alloc nomem;
        _RAISE(nomem);
    }

    return res;
}


這樣Crt會(huì)把此次分配內(nèi)存的文件名和行號(hào)以及大小等記錄下來(lái),最后當(dāng)調(diào)用用_CrtDumpMemoryLeaks(); 時(shí)如果還沒(méi)釋放就會(huì)打印出來(lái)。
結(jié)果如下:
復(fù)制代碼 代碼如下:

Detected memory leaks!
Dumping objects ->
f:\test\memleakchecker\memleakchecker\memleakchecker.cpp(23) : {108} normal block at 0x0003A1A8, 10 bytes long.
 Data: <          > CD CD CD CD CD CD CD CD CD CD
f:\test\memleakchecker\memleakchecker\memleakchecker.cpp(22) : {107} client block at 0x0003A160, subtype 0, 10 bytes long.
 Data: <          > CD CD CD CD CD CD CD CD CD CD
f:\test\memleakchecker\memleakchecker\memleakchecker.cpp(21) : {106} client block at 0x0003A120, subtype 0, 1 bytes long.
 Data: < > 00
Object dump complete.

下面是一些注意事項(xiàng):
(1) #define _CRTDBG_MAP_ALLOC 的作用
如果不定義這個(gè)宏, C方式的malloc泄露不會(huì)被記錄下來(lái)。

(2)數(shù)字{108} {107}的作用
表示第幾次分配, 你可以通過(guò)_CrtSetBreakAlloc程序運(yùn)行到預(yù)定次數(shù)時(shí)暫停 ,比如

復(fù)制代碼 代碼如下:

int _tmain(int argc, _TCHAR* argv[])
{
    _CrtSetBreakAlloc(108);

    char* p = new char();
    char* pp = new char[10];
    char* ppp = (char*)malloc(10);

    _CrtDumpMemoryLeaks();

    return 0;
}


(3)如果程序有多個(gè)出口或是有涉及到全局變量, 可以通過(guò)_CrtSetDbgFlag 設(shè)置標(biāo)志讓程序退出時(shí)自動(dòng)打印泄露 , 比如
復(fù)制代碼 代碼如下:

int _tmain(int argc, _TCHAR* argv[])
{
    _CrtSetDbgFlag ( _CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF );

    char* p = new char();
    char* pp = new char[10];
    char* ppp = (char*)malloc(10);

    return 0;
}


(4)我們知道宏替代是最粗暴的方式, 所以盡量把下面new的替代宏放到每個(gè)Cpp里而不是放到一個(gè)通用的頭文件中, 實(shí)際上MFC也是這么做的
復(fù)制代碼 代碼如下:

#ifdef _DEBUG
#define new DEBUG_CLIENTBLOCK
#endif

(5)上面的operator new只能照顧到最普通的new, 實(shí)際上operator new是有任意多種重載方式, 只需要確保第一個(gè)參數(shù)是表示大小。 比如下面的placement new就會(huì)編譯失敗, 因?yàn)楹晏娲蟾袷讲环弦罅耍?所以如果你的CPP用了非標(biāo)準(zhǔn)的new, 就不要加入new的檢測(cè)宏了。
復(fù)制代碼 代碼如下:

#include <new>

#ifdef _DEBUG
#define DEBUG_CLIENTBLOCK   new( _CLIENT_BLOCK, __FILE__, __LINE__)
#else
#define DEBUG_CLIENTBLOCK
#endif
#define _CRTDBG_MAP_ALLOC
#include <crtdbg.h>
#ifdef _DEBUG
#define new DEBUG_CLIENTBLOCK
#endif


int _tmain(int argc, _TCHAR* argv[])
{
    _CrtSetDbgFlag ( _CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF );

    char* p = new char();
    char* pp = new char[10];
    char* ppp = (char*)malloc(10);

    char d;
    char* p1 = new(&d) char('a');

    return 0;
}


(6)因?yàn)镾TL里map內(nèi)的tree用到了placement new,  所以如果你這樣用會(huì)編譯失?。?BR>
復(fù)制代碼 代碼如下:

#ifdef _DEBUG
#define DEBUG_CLIENTBLOCK   new( _CLIENT_BLOCK, __FILE__, __LINE__)
#else
#define DEBUG_CLIENTBLOCK
#endif
#define _CRTDBG_MAP_ALLOC
#include <crtdbg.h>
#ifdef _DEBUG
#define new DEBUG_CLIENTBLOCK
#endif

#include <map>


你應(yīng)該把 #include <map>放到 宏定義的前面。

(7) 如果你在宏 #define new DEBUG_CLIENTBLOCK 之后再聲明或定義 operator new函數(shù), 都會(huì)因?yàn)楹晏娲幾g失敗。
而STL的xdebug文件恰恰申明了operator new函數(shù), 所以請(qǐng)確保new的替代宏放在所有include頭文件的最后, 尤其要放在STL頭文件的后面。

復(fù)制代碼 代碼如下:

//MyClass.cpp
#include "myclass.h"
#include <map>
#include <algorithm>

#ifdef _DEBUG
#define new DEBUG_CLIENTBLOCK
#endif

MyClass::MyClass()
{
    char* p = new char('a');
}


(8)如果你覺(jué)得上面的這種new替代宏分散在各個(gè)CPP里太麻煩, 想把所有的東西放到一個(gè)通用頭文件里,請(qǐng)參考下面定義的方式:
復(fù)制代碼 代碼如下:

//MemLeakChecker.h
#include <map>
#include <algorithm>
//other STL file

#ifdef _DEBUG
#define DEBUG_CLIENTBLOCK   new( _CLIENT_BLOCK, __FILE__, __LINE__)
#else
#define DEBUG_CLIENTBLOCK
#endif
#define _CRTDBG_MAP_ALLOC
#include <crtdbg.h>
#ifdef _DEBUG
#define new DEBUG_CLIENTBLOCK
#endif


(9)簡(jiǎn)單判斷某個(gè)獨(dú)立函數(shù)有沒(méi)有內(nèi)存泄露可以用下面的方法:
復(fù)制代碼 代碼如下:

class DbgMemLeak
{
    _CrtMemState m_checkpoint;

public:
    explicit DbgMemLeak()
    {  
        _CrtMemCheckpoint(&m_checkpoint);
    };

    ~DbgMemLeak()
    {
        _CrtMemState checkpoint;
        _CrtMemCheckpoint(&checkpoint);
        _CrtMemState diff;
        _CrtMemDifference(&diff, &m_checkpoint, &checkpoint);
        _CrtMemDumpStatistics(&diff);
        _CrtMemDumpAllObjectsSince(&diff);
    };
};


int _tmain(int argc, _TCHAR* argv[])
{
    DbgMemLeak check;
    {
        char* p = new char();
        char* pp = new char[10];
        char* ppp = (char*)malloc(10);
    }

    return 0;
}


(10) 其實(shí)知道了原理, 自己寫(xiě)一套C++內(nèi)存泄露檢測(cè)也不難, 主要是重載operator new和operator delete, 可以把每次內(nèi)存分配情況都記錄在一個(gè)Map里, delete時(shí)刪除記錄, 最后程序退出時(shí)把map里沒(méi)有delete的打印出來(lái)。 當(dāng)然我們知道Crt在實(shí)現(xiàn)new時(shí)一般實(shí)際上調(diào)的是malloc, 而malloc可能又是調(diào)HeapAlloc,而HeapAlloc可能又是調(diào)用RtlAllocateHeap, 所以理論上我們可以在這些函數(shù)的任意一層攔截和記錄。但是如果你要實(shí)現(xiàn)自己的跨平臺(tái)內(nèi)存泄露檢測(cè),還是重載operator new吧。

相關(guān)文章

  • 詳解C++編程中運(yùn)算符的使用

    詳解C++編程中運(yùn)算符的使用

    這篇文章主要介紹了詳解C++編程中運(yùn)算符的使用,是C++入門(mén)學(xué)習(xí)中的基礎(chǔ)知識(shí),需要的朋友可以參考下
    2015-09-09
  • OpenCV利用背景建模檢測(cè)運(yùn)動(dòng)物體

    OpenCV利用背景建模檢測(cè)運(yùn)動(dòng)物體

    這篇文章主要為大家詳細(xì)介紹了OpenCV利用背景建模檢測(cè)運(yùn)動(dòng)物體,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-01-01
  • FFmpeg進(jìn)階教程之給視頻添加文字水印

    FFmpeg進(jìn)階教程之給視頻添加文字水印

    FFmpeg是一套可以用來(lái)記錄、轉(zhuǎn)換數(shù)字音頻、視頻,并能將其轉(zhuǎn)化為流的開(kāi)源計(jì)算機(jī)程序,下面這篇文章主要給大家介紹了關(guān)于FFmpeg進(jìn)階教程之給視頻添加文字水印的相關(guān)資料,需要的朋友可以參考下
    2022-11-11
  • MFC對(duì)話(huà)框中添加狀態(tài)欄的方法

    MFC對(duì)話(huà)框中添加狀態(tài)欄的方法

    這篇文章主要介紹了MFC對(duì)話(huà)框中添加狀態(tài)欄的方法,實(shí)例分析了MFC對(duì)話(huà)框添加狀態(tài)欄所涉及的相關(guān)成員變量與事件實(shí)現(xiàn)技巧,需要的朋友可以參考下
    2015-07-07
  • c語(yǔ)言之char*和unsigned?char*的區(qū)別及說(shuō)明

    c語(yǔ)言之char*和unsigned?char*的區(qū)別及說(shuō)明

    這篇文章主要介紹了c語(yǔ)言之char*和unsigned?char*的區(qū)別及說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-08-08
  • C/C++哈希表優(yōu)化LeetCode題解997找到小鎮(zhèn)的法官

    C/C++哈希表優(yōu)化LeetCode題解997找到小鎮(zhèn)的法官

    這篇文章主要為大家介紹了C/C++哈希表優(yōu)化題解997找到小鎮(zhèn)的法官示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-12-12
  • 用C語(yǔ)言完整實(shí)現(xiàn)2048游戲

    用C語(yǔ)言完整實(shí)現(xiàn)2048游戲

    2048是一款數(shù)字益智游戲,初始數(shù)字則是由2+2組成的基數(shù)4。在操作方面的不同則表現(xiàn)為一步一格的移動(dòng),變成更為爽快的一次到底。相同數(shù)字的方?jīng)r在靠攏、相撞時(shí)會(huì)相加。系統(tǒng)給予的數(shù)字方塊不是2就是4,玩家要想辦法在這小小的16格范圍中湊出2048這個(gè)數(shù)字方塊
    2021-11-11
  • C語(yǔ)言實(shí)現(xiàn)通用數(shù)據(jù)結(jié)構(gòu)之通用映射(HashMap)

    C語(yǔ)言實(shí)現(xiàn)通用數(shù)據(jù)結(jié)構(gòu)之通用映射(HashMap)

    這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)通用數(shù)據(jù)結(jié)構(gòu)之通用映射,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-11-11
  • C++類(lèi)和對(duì)象之多態(tài)詳解

    C++類(lèi)和對(duì)象之多態(tài)詳解

    大家好,本篇文章主要講的是C++類(lèi)和對(duì)象之多態(tài)詳解,感興趣的同學(xué)趕快來(lái)看一看吧,對(duì)你有幫助的話(huà)記得收藏一下,方便下次瀏覽
    2021-12-12
  • c語(yǔ)言單詞本的新增、刪除、查詢(xún)按順序顯示功能

    c語(yǔ)言單詞本的新增、刪除、查詢(xún)按順序顯示功能

    這篇文章主要介紹了c語(yǔ)言單詞本的新增、刪除、查詢(xún)按順序顯示功能,本文通過(guò)示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-12-12

最新評(píng)論