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

            鍵盤的詠嘆調(diào)

            常用鏈接

            統(tǒng)計(jì)

            最新評(píng)論

            【zz】使用CRT調(diào)試內(nèi)存分配堆來找出未釋放的內(nèi)存空間

               原地址:http://blog.csdn.net/Donjuan/archive/2009/02/02/3859154.aspx

               忘記釋放已經(jīng)分配的內(nèi)存是一種常見的編程錯(cuò)誤,當(dāng)然我指的是在
            C++編程當(dāng)中,例如下面的代碼里面就存在一個(gè)忘記釋放內(nèi)存的編程錯(cuò)誤。我個(gè)人覺得忘記釋放內(nèi)存的編程錯(cuò)誤是不可避免的,畢竟程序員都是人,困了,心情不好了,代碼過于復(fù)雜啦等等都可能導(dǎo)致忘記加上一句delete XXX語句。

            // 未釋放內(nèi)存空間.cpp : Defines the entry point for the console application.

             

            #include "stdafx.h"

            #include <windows.h>

            #include <string>

            #include <iostream>

             

            using namespace std;

             

            class CTestClass

            {

            public:

                CTestClass(LPWSTR szName)

                {

                    m_lpName = new wstring(szName);

                }

             

                ~CTestClass()

                {

                }

             

                void PrintName()

                {

                    wcout << *m_lpName << endl;

                }

             

            private:

                wstring *m_lpName;

            };

             

            HRESULT CreateTestClass(LPWSTR szName, CTestClass **ppObject)

            {

                *ppObject = new CTestClass(szName);

                if ( (*ppObject) == NULL )

                    return E_FAIL;

                else

                    return S_OK;

            }

             

            int _tmain(int argc, _TCHAR* argv[])

            {

                CTestClass *pObject = NULL;

                HRESULT hr = CreateTestClass(L"This is a Test", &pObject);

                if ( hr != S_OK )

                {

                    return -1;

                }

                else

                {

                    pObject->PrintName();  // pObject沒有被釋放

                    return 0;

                }

            }

             

            實(shí)際上Visual Studio已經(jīng)提供了方法幫助你快速找到這些沒有釋放的內(nèi)存。

             

            Visual Studio提供了一系列的CRT調(diào)試APICRT提供了一個(gè)調(diào)試內(nèi)存分配堆,可以跟蹤和管理內(nèi)存在什么地方分配,當(dāng)你在這個(gè)堆上分配內(nèi)存的時(shí)候,每一次內(nèi)存分配調(diào)用例如malloc或者newCRT都會(huì)額外分配大約36個(gè)字節(jié)用來保存例如這個(gè)內(nèi)存塊分配的文件名、行號(hào)、內(nèi)存塊的大小等信息,最后CRT將這些內(nèi)存塊使用一個(gè)雙鏈表鏈接起來。每一次內(nèi)存釋放的時(shí)候,free或者delete函數(shù)就從這個(gè)內(nèi)存塊鏈表里面將要釋放的內(nèi)存塊刪除,因此在需要檢查內(nèi)存泄漏的時(shí)候,只要遍歷這個(gè)雙鏈表依次打印出這些內(nèi)存塊就可以發(fā)現(xiàn)所有未釋放的內(nèi)存了。下面是CRT內(nèi)存塊的原始聲明:

            typedef struct _CrtMemBlockHeader

            {

                // Pointer to the block allocated just before this one:

                struct _CrtMemBlockHeader *pBlockHeaderNext;

                // Pointer to the block allocated just after this one:

                struct _CrtMemBlockHeader *pBlockHeaderPrev;

                char *szFileName;    // File name

                int nLine;           // Line number

                size_t nDataSize;    // Size of user block

                int nBlockUse;       // Type of block

                long lRequest;       // Allocation number

                // Buffer just before (lower than) the user's memory:

                unsigned char gap[nNoMansLandSize];

            } _CrtMemBlockHeader;

             

            下面的代碼演示了如何使用CRT提供的調(diào)試API來修改剛才的源文件檢測未釋放的內(nèi)存空間(注意紅色添加的部分):

            // 未釋放內(nèi)存空間.cpp : Defines the entry point for the console application.

            //

             

            #include "stdafx.h"

            #include <windows.h>

            #include <string>

            #include <iostream>

            // 使用CRT調(diào)試API

            #include <crtdbg.h>

             

            using namespace std;

             

            // 將所有的內(nèi)存分配函數(shù)new替換成CRT提供的調(diào)試new

            #ifdef _DEBUG

            #   define DEBUG_CLIENTBLOCK new(_CLIENT_BLOCK, __FILE__, __LINE__)

            #else

            #   define DEBUG_CLIENTBLOCK

            #endif

             

            #ifdef _DEBUG

            #    define new DEBUG_CLIENTBLOCK

            #endif

             

            class CTestClass

            {

            public:

                CTestClass(LPWSTR szName)

                {

                    m_lpName = new wstring(szName);

                }

             

                ~CTestClass()

                {

                }

             

                void PrintName()

                {

                    wcout << *m_lpName << endl;

                }

             

            private:

                wstring *m_lpName;

            };

             

            HRESULT CreateTestClass(LPWSTR szName, CTestClass **ppObject)

            {

                *ppObject = new CTestClass(szName);

                if ( (*ppObject) == NULL )

                    return E_FAIL;

                else

                    return S_OK;

            }

             

            int _tmain(int argc, _TCHAR* argv[])

            {  

            // 設(shè)置CRT調(diào)試API的報(bào)表輸出模式,將所有的錯(cuò)誤、警告還有斷言都輸出到控制臺(tái)

                _CrtSetReportMode( _CRT_WARN, _CRTDBG_MODE_FILE );

                _CrtSetReportFile( _CRT_WARN, _CRTDBG_FILE_STDOUT );

                _CrtSetReportMode( _CRT_ERROR, _CRTDBG_MODE_FILE );

                _CrtSetReportFile( _CRT_ERROR, _CRTDBG_FILE_STDOUT );

                _CrtSetReportMode( _CRT_ASSERT, _CRTDBG_MODE_FILE );

                _CrtSetReportFile( _CRT_ASSERT, _CRTDBG_FILE_STDOUT );

             

                CTestClass *pObject = NULL;

                HRESULT hr = CreateTestClass(L"This is a Test", &pObject);

                if ( hr != S_OK )

                {

                    return -1;

                }

                else

                {

                    pObject->PrintName();

             

            // 檢查未釋放的內(nèi)存

                    _CrtDumpMemoryLeaks();

                    return 0;

                }

            }

            posted on 2009-02-03 11:59 鍵盤的詠嘆調(diào) 閱讀(417) 評(píng)論(0)  編輯 收藏 引用 所屬分類: C++

            日韩美女18网站久久精品| 久久免费看黄a级毛片| 嫩草伊人久久精品少妇AV| 久久伊人精品一区二区三区| 亚洲午夜久久久久久久久电影网| 99久久国产宗和精品1上映| 精品久久无码中文字幕| 国内精品欧美久久精品| 伊人久久久AV老熟妇色| 久久99精品国产99久久6男男| 久久久中文字幕日本| 久久精品国产第一区二区三区| 99热热久久这里只有精品68| 国产Av激情久久无码天堂| 久久精品女人天堂AV麻| 久久国产亚洲精品无码| 精品人妻伦一二三区久久| 久久国产免费观看精品3| 色婷婷狠狠久久综合五月| 成人久久综合网| 久久精品国产亚洲AV不卡| 国产精品成人久久久久三级午夜电影 | 久久精品国产99国产精品澳门| 一本久道久久综合狠狠躁AV| A级毛片无码久久精品免费| 久久久久久亚洲精品成人 | 超级97碰碰碰碰久久久久最新| 国产欧美久久久精品| 久久人妻少妇嫩草AV无码专区| 区亚洲欧美一级久久精品亚洲精品成人网久久久久 | 国产三级精品久久| 国产综合久久久久久鬼色| 久久www免费人成看片| 香蕉久久久久久狠狠色| 精品国产青草久久久久福利 | 国产成人精品白浆久久69| 久久久www免费人成精品| 婷婷国产天堂久久综合五月| 色综合久久久久综合99| 麻豆久久久9性大片| 久久亚洲中文字幕精品一区|