亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C++ > 正文

C++中關于Crt的內存泄漏檢測的分析介紹

2020-01-26 16:18:54
字體:
來源:轉載
供稿:網友

盡管這個概念已經讓人說濫了 ,還是想簡單記錄一下, 以備以后查詢。

復制代碼 代碼如下:

#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;
}


主要原理是運用Crt 的內存調試功能, 通過宏替代默認的operator new, 讓它被下面版本替代:
復制代碼 代碼如下:

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會把此次分配內存的文件名和行號以及大小等記錄下來,最后當調用用_CrtDumpMemoryLeaks(); 時如果還沒釋放就會打印出來。
結果如下:
復制代碼 代碼如下:

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.

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

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

復制代碼 代碼如下:

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)如果程序有多個出口或是有涉及到全局變量, 可以通過_CrtSetDbgFlag 設置標志讓程序退出時自動打印泄露 , 比如
復制代碼 代碼如下:

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的替代宏放到每個Cpp里而不是放到一個通用的頭文件中, 實際上MFC也是這么做的
復制代碼 代碼如下:

#ifdef _DEBUG
#define new DEBUG_CLIENTBLOCK
#endif

(5)上面的operator new只能照顧到最普通的new, 實際上operator new是有任意多種重載方式, 只需要確保第一個參數是表示大小。 比如下面的placement new就會編譯失敗, 因為宏替代后格式不符合要求了, 所以如果你的CPP用了非標準的new, 就不要加入new的檢測宏了。
復制代碼 代碼如下:

#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)因為STL里map內的tree用到了placement new,  所以如果你這樣用會編譯失?。?BR>
復制代碼 代碼如下:

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


你應該把 #include <map>放到 宏定義的前面。

(7) 如果你在宏 #define new DEBUG_CLIENTBLOCK 之后再聲明或定義 operator new函數, 都會因為宏替代而編譯失敗。
而STL的xdebug文件恰恰申明了operator new函數, 所以請確保new的替代宏放在所有include頭文件的最后, 尤其要放在STL頭文件的后面。

復制代碼 代碼如下:

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

#ifdef _DEBUG
#define new DEBUG_CLIENTBLOCK
#endif

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


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

//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)簡單判斷某個獨立函數有沒有內存泄露可以用下面的方法:
復制代碼 代碼如下:

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) 其實知道了原理, 自己寫一套C++內存泄露檢測也不難, 主要是重載operator new和operator delete, 可以把每次內存分配情況都記錄在一個Map里, delete時刪除記錄, 最后程序退出時把map里沒有delete的打印出來。 當然我們知道Crt在實現new時一般實際上調的是malloc, 而malloc可能又是調HeapAlloc,而HeapAlloc可能又是調用RtlAllocateHeap, 所以理論上我們可以在這些函數的任意一層攔截和記錄。但是如果你要實現自己的跨平臺內存泄露檢測,還是重載operator new吧。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲人成在线观看网站高清| 欧美一级片免费在线| 91av在线影院| 91国内产香蕉| 国产成人久久久精品一区| 中文字幕国产精品| 欧美夫妻性生活视频| x99av成人免费| 国产精品流白浆视频| 国产精品扒开腿做爽爽爽视频| 午夜精品国产精品大乳美女| 国产精品激情av电影在线观看| 国产日韩视频在线观看| 久久国产加勒比精品无码| 欧美一性一乱一交一视频| 国产热re99久久6国产精品| 欧美肥老太性生活视频| 激情久久av一区av二区av三区| 亚洲精品999| 日韩亚洲欧美中文在线| 欧美视频在线观看 亚洲欧| 成人444kkkk在线观看| 国产精品美腿一区在线看| 国产精品久久久久久av| 一区二区三区国产在线观看| 欧美性xxxx极品hd满灌| 亚洲国产中文字幕久久网| 亚洲自拍偷拍区| 国产精品免费久久久久影院| 亚洲一区av在线播放| 97超碰蝌蚪网人人做人人爽| 久久精品中文字幕免费mv| 亚洲第一精品夜夜躁人人躁| 久久久亚洲成人| 日韩视频永久免费观看| 亚洲国产精品久久久久秋霞不卡| 欧美高跟鞋交xxxxxhd| 久久五月天综合| 7777kkkk成人观看| 成人免费在线网址| 欧美在线一级视频| 91免费看片在线| 国模叶桐国产精品一区| 色综合久久精品亚洲国产| 中文字幕视频在线免费欧美日韩综合在线看| 97超碰蝌蚪网人人做人人爽| 久久久久久久影院| 日韩va亚洲va欧洲va国产| 国产丝袜一区二区三区| 国产精品一区二区久久久| 亚洲www在线观看| 久久国产精品亚洲| 亚洲欧美国产制服动漫| 欧美精品一区二区三区国产精品| 国产亚洲欧美一区| 国产精品男女猛烈高潮激情| 国内精品模特av私拍在线观看| 亚洲成人动漫在线播放| 91色琪琪电影亚洲精品久久| 成人激情视频免费在线| 91精品国产综合久久香蕉的用户体验| 久久精品久久精品亚洲人| 欧美性猛交xxx| 亚洲性生活视频在线观看| 国产在线观看精品| 欧美精品亚州精品| 中文字幕自拍vr一区二区三区| 欧美日韩国产中文精品字幕自在自线| 26uuu日韩精品一区二区| 久久视频在线看| 亚洲第一国产精品| 国产精品∨欧美精品v日韩精品| 成人免费看黄网站| 久久天天躁狠狠躁夜夜躁2014| 黑人与娇小精品av专区| 奇米影视亚洲狠狠色| 国产精品高潮呻吟久久av无限| 欧美最顶级的aⅴ艳星| 中文字幕日韩av| 91麻豆国产语对白在线观看| 国产精品久久久久久久av大片| 国产专区欧美专区| 日韩av大片免费看| 国产视频精品xxxx| 欧美在线国产精品| 欧美人成在线视频| 91香蕉嫩草神马影院在线观看| 91精品国产91久久久久| 欧美精品videosex牲欧美| 亚洲第一二三四五区| 国产在线拍偷自揄拍精品| 亚洲精品乱码久久久久久按摩观| 欧美日韩亚洲高清| 色综合视频网站| 国产精品白丝av嫩草影院| 日韩免费高清在线观看| 国产亚洲免费的视频看| 国模精品视频一区二区| 国产999视频| 国产精品揄拍一区二区| 国产精品视频在线观看| 亚洲欧美国产精品久久久久久久| 91在线观看免费观看| 国内精品久久久久| 国产亚洲精品美女久久久久| 伊人av综合网| 97视频色精品| 亚洲www永久成人夜色| 影音先锋日韩有码| 久久精品91久久香蕉加勒比| 成人欧美在线观看| 国产一区二区三区精品久久久| 国产精品极品尤物在线观看| 久久久成人的性感天堂| 一区二区三区视频免费在线观看| 欧美激情视频网站| 亚洲欧美中文字幕| 国产精品久久久久久久久免费看| 自拍偷拍亚洲欧美| 欧美黑人极品猛少妇色xxxxx| 国产精品久久久久久久7电影| 秋霞av国产精品一区| 奇米四色中文综合久久| 高清日韩电视剧大全免费播放在线观看| 欧美电影在线观看| 2019中文字幕在线| 91精品国产高清久久久久久久久| 69久久夜色精品国产69乱青草| 91精品国产高清自在线| 日韩欧美高清在线视频| 在线观看欧美日韩国产| 免费不卡欧美自拍视频| 欧美日韩福利视频| 日韩av电影手机在线观看| 色偷偷亚洲男人天堂| 亚洲综合大片69999| 亚洲精品av在线| 精品日韩中文字幕| 国产精品免费小视频| 国产一区香蕉久久| 亚洲第一区在线观看| 亚洲国产精品久久91精品| 国产欧美精品一区二区三区介绍| 欧美黑人又粗大| 96精品久久久久中文字幕| 久久综合伊人77777尤物| 国内揄拍国内精品少妇国语| 日韩精品有码在线观看| 日本精品一区二区三区在线| 久久人人爽人人| 中文字幕欧美国内| 91免费观看网站| 久久久视频精品| 九色成人免费视频| 136fldh精品导航福利| 日韩高清人体午夜| 亚洲影视中文字幕| 欧美成人剧情片在线观看| 中文字幕亚洲一区在线观看| 国产精品99久久久久久久久| 国产美女久久精品香蕉69| 国产亚洲一区二区在线| 欧美性xxxx| 欧美日韩精品中文字幕|