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

            C++ Programmer's Cookbook

            {C++ 基礎(chǔ)} {C++ 高級(jí)} {C#界面,C++核心算法} {設(shè)計(jì)模式} {C#基礎(chǔ)}

            C++內(nèi)存泄露的檢測(cè)(三)



            一 簡(jiǎn)單的對(duì)內(nèi)存的分配和釋放跟蹤,并將結(jié)果輸出到console,它也是一般C++內(nèi)存泄露的檢測(cè)原理,來(lái)自C++編程思想:
            (比較簡(jiǎn)單,大家都可以看的明白的哦)實(shí)現(xiàn)如下:

            MemCheck.h
            //: C02:MemCheck.h
            #ifndef MEMCHECK_H
            #define MEMCHECK_H
            #include 
            <cstddef>  // for size_t

            // Hijack the new operator (both scalar and array versions)
            void* operator new(std::size_t, const char*long);
            void* operator new[](std::size_t, const char*long);
            #define new new (__FILE__, __LINE__)

            extern bool traceFlag;
            #define TRACE_ON() traceFlag = true
            #define TRACE_OFF() traceFlag = false

            extern bool activeFlag;
            #define MEM_ON() activeFlag = true
            #define MEM_OFF() activeFlag = false

            #endif
            ///:~

            MemCheck.cpp
            //: C02:MemCheck.cpp {O}
            #include <cstdio>
            #include 
            <cstdlib>
            #include 
            <cassert>
            using namespace std;
            #undef new

            // Global flags set by macros in MemCheck.h
            bool traceFlag = true;
            bool activeFlag = false;

            namespace {

            // Memory map entry type
            struct Info {
              
            void* ptr;
              
            const char* file;
              
            long line;
            }
            ;

            // Memory map data
            const size_t MAXPTRS = 10000u;
            Info memMap[MAXPTRS];
            size_t nptrs 
            = 0;

            // Searches the map for an address
            int findPtr(void* p) 
            {
              
            for (int i = 0; i < nptrs; ++i)
                
            if (memMap[i].ptr == p)
                  
            return i;
              
            return -1;
            }


            void delPtr(void* p)
            {
              
            int pos = findPtr(p);
              assert(p 
            >= 0);
              
            // Remove pointer from map
              for (size_t i = pos; i < nptrs-1++i)
                memMap[i] 
            = memMap[i+1];
              
            --nptrs;
            }


            // Dummy type for static destructor
            struct Sentinel 
            {
              
            ~Sentinel()
              
            {
                
            if (nptrs > 0
                
            {
                  printf(
            "Leaked memory at:\n");
                  
            for (size_t i = 0; i < nptrs; ++i)
                    printf(
            "\t%p (file: %s, line %ld)\n",
                      memMap[i].ptr, memMap[i].file, memMap[i].line);
                }

                
            else
                  printf(
            "No user memory leaks!\n");
              }

            }
            ;

            // Static dummy object
            Sentinel s;

            }
             // End anonymous namespace

            // Overload scalar new
            void* operator new(size_t siz, const char* file,
              
            long line) {
              
            void* p = malloc(siz);
              
            if (activeFlag)
              
            {
                
            if (nptrs == MAXPTRS)
                
            {
                  printf(
            "memory map too small (increase MAXPTRS)\n");
                  exit(
            1);
                }

                memMap[nptrs].ptr 
            = p;
                memMap[nptrs].file 
            = file;
                memMap[nptrs].line 
            = line;
                
            ++nptrs;
              }

              
            if (traceFlag) 
              
            {
                printf(
            "Allocated %u bytes at address %p ", siz, p);
                printf(
            "(file: %s, line: %ld)\n", file, line);
              }

              
            return p;
            }


            // Overload array new
            void* operator new[](size_t siz, const char* file,
              
            long line) {
              
            return operator new(siz, file, line);
            }


            // Override scalar delete
            void operator delete(void* p) 
            {
              
            if (findPtr(p) >= 0
              
            {
                free(p);
                assert(nptrs 
            > 0);
                delPtr(p);
                
            if (traceFlag)
                  printf(
            "Deleted memory at address %p\n", p);
              }

              
            else if (!&& activeFlag)
                printf(
            "Attempt to delete unknown pointer: %p\n", p);
            }


            // Override array delete
            void operator delete[](void* p) {
              
            operator delete(p);
            }
             ///:~


            二 說(shuō)明:
            1)通過(guò)重載new和delete來(lái)實(shí)現(xiàn)
            2)使用時(shí)需要在工程中加入MemCheck.h和MemCheck.cpp,在需要檢測(cè)的文件的前面include “MemCheck.h”,但是必須在所有的include的最后。
            3)MEM_ON(),MEM_OFF()用來(lái)打開或關(guān)閉檢測(cè)
            4)TRACE_ON(),和TRACE_OFF()用來(lái)打開或關(guān)閉檢測(cè)結(jié)果的輸出
            5)可以檢測(cè)代碼中使用了流,標(biāo)準(zhǔn)容器,以及某個(gè)類的構(gòu)造函數(shù)分配了空間

            三 使用實(shí)例:
            console 的project中加入下面的file:
            // MemoryLeak3.cpp : Defines the entry point for the console application.
            //

            #include 
            <iostream>
            #include 
            <vector>
            #include 
            <cstring>

            #include 
            "MemCheck.h"   // Must appear last!
            using namespace std;

            void Test()
            {
                
            int *= new int(0);
            }


            class MyClass
            {
            private:
                
            int *p;
            public:
                MyClass()
                
            {
                    
            if(p != NULL)
                    
            {
                        p 
            = new int(0);
                    }

                }

                
            ~MyClass()
                
            {
                    
            if(p != NULL)
                    
            {
                        delete p;
                        p 
            = NULL;
                    }

                }

            }
            ;

            void Test2()
            {
                
            int *= NULL; // better for read
                    i = new int(0);    
                    
            int *&= i; // pointer's reference
                    delete i;

                    MyClass 
            *pMyClass = new MyClass();

                    std::vector
            <MyClass*> myClasses;
                    myClasses.push_back(
            new MyClass());
                    myClasses.push_back(
            new MyClass());

                    std::vector
            <void*> myVector;
                    myVector.push_back(
            new MyClass());
                    myVector.push_back(
            new MyClass());
                    delete (MyClass 
            *)(myVector.at(0));
                    delete myVector.at(
            1); // memory leak
            }


            class Foo 
            {
              
            char* s;
            public:
              Foo(
            const char*s ) 
              
            {
                
            this->= new char[strlen(s) + 1];
                strcpy(
            this->s, s);
              }

              
            ~Foo() 
              
            {
                delete [] s;
              }

            }
            ;
            void Test3()
            {
                cout 
            << "hello\n";
                
            int* p = new int;
                delete p;
                
            int* q = new int[3];
                delete [] q;
                
            int* r;
                
            /*delete r;*/
                vector
            <int> v;
                v.push_back(
            1);
                Foo s(
            "goodbye");
            }

            int main() 
            {
                TRACE_OFF();
                MEM_ON();
                Test();
                Test2();
                Test3();
                MEM_OFF();
            }
             ///:~

            四 測(cè)試結(jié)果如下:


            posted on 2007-07-23 18:28 夢(mèng)在天涯 閱讀(5439) 評(píng)論(6)  編輯 收藏 引用 所屬分類: CPlusPlus

            評(píng)論

            # re: C++內(nèi)存泄露檢測(cè)(三) 2007-07-23 18:34 夢(mèng)在天涯

            那位能夠解釋一下:#define new new (__FILE__, __LINE__)
              回復(fù)  更多評(píng)論   

            # re: C++內(nèi)存泄露檢測(cè)(三) 2007-07-23 21:46 pass86

            我怎么沒(méi)行得通呢,用了mingw,gcc3.4.4  回復(fù)  更多評(píng)論   

            # re: C++內(nèi)存泄露檢測(cè)(三) 2007-07-23 21:55 萬(wàn)連文

            最簡(jiǎn)單有效的辦法,boundchecker。其他的方法都很乏力,不能保證其他人和自己一致采用一種檢查辦法。  回復(fù)  更多評(píng)論   

            # re: C++內(nèi)存泄露檢測(cè)(三) 2007-07-24 08:34 jazz

            // Override array delete
            void operator delete[](void* p) {
            operator delete(p);
            } /**////:~

            這段代碼看起來(lái)本身就要泄漏資源了...只要p指向的不是內(nèi)置數(shù)據(jù)類型 vs下就會(huì)泄漏內(nèi)存  回復(fù)  更多評(píng)論   

            # re: C++內(nèi)存泄露檢測(cè)(三) 2007-07-24 08:49 jazz

            漏看了 new[]的實(shí)現(xiàn),總覺(jué)得不會(huì)做初始化啊...  回復(fù)  更多評(píng)論   

            # re: C++內(nèi)存泄露檢測(cè)(三) 2007-07-24 15:46 深邃者

            for (int i = 0; i < nptrs; ++i)
            這里nptrs=0啊 是不是寫錯(cuò)了?
            還有這里:
            for (size_t i = pos; i < nptrs-1; ++i)
            memMap[i] = memMap[i+1];
            --nptrs;
              回復(fù)  更多評(píng)論   

            公告

            EMail:itech001#126.com

            導(dǎo)航

            統(tǒng)計(jì)

            • 隨筆 - 461
            • 文章 - 4
            • 評(píng)論 - 746
            • 引用 - 0

            常用鏈接

            隨筆分類

            隨筆檔案

            收藏夾

            Blogs

            c#(csharp)

            C++(cpp)

            Enlish

            Forums(bbs)

            My self

            Often go

            Useful Webs

            Xml/Uml/html

            搜索

            •  

            積分與排名

            • 積分 - 1804663
            • 排名 - 5

            最新評(píng)論

            閱讀排行榜

            伊人久久大香线蕉av不卡 | 久久青青草原综合伊人| 久久亚洲国产精品成人AV秋霞 | 国产女人aaa级久久久级| 久久这里只有精品视频99| 久久精品亚洲AV久久久无码| 久久久精品人妻一区二区三区蜜桃 | 91精品国产综合久久四虎久久无码一级| 久久香蕉国产线看观看99| 日本精品久久久久久久久免费| 日韩人妻无码精品久久免费一| 久久精品国产99国产精品澳门 | 久久免费视频观看| 久久久久久国产精品无码下载 | 午夜精品久久久久久| 精品久久久久中文字幕日本| 久久午夜综合久久| 免费观看久久精彩视频| 久久亚洲AV无码精品色午夜| 久久99国产精品成人欧美| 久久精品国产清高在天天线| 色综合久久夜色精品国产| 亚洲国产精品久久久久婷婷老年| 久久人妻无码中文字幕| 久久高清一级毛片| 亚洲狠狠综合久久| .精品久久久麻豆国产精品| 亚洲国产精品无码久久一线| 日韩va亚洲va欧美va久久| 国产精品美女久久久久av爽 | 久久99精品久久久久子伦| 久久久久国产精品人妻 | 久久91综合国产91久久精品| 色偷偷偷久久伊人大杳蕉| 久久婷婷国产剧情内射白浆| 久久人人爽人人精品视频| 久久精品国产亚洲7777| 久久久久一本毛久久久| 久久久久亚洲精品男人的天堂| 久久久久亚洲精品无码网址| 久久精品国产精品亚洲艾草网美妙|