青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

春暖花開
雪化了,花開了,春天來了
posts - 149,comments - 125,trackbacks - 0
 

檢測內存泄露的工具:debugnew
http://dev.csdn.net/article/58/58407.shtm

網上有一個流傳甚廣的檢測內存泄露的工具:debugnew(debugnew.h/debugnew.cpp)
用法很簡單,把debugnew.cpp放在項目里一起編譯,需要檢測的文件把debugnew.h嵌在文件的最前面。

為方便使用,對源代碼做了一些小的改動。

下面是一些簡單的說明:

1、new 的重載
void* operator new (size_t size, const char* file, int line);        ⑴
void* operator new [] (size_t size, const char*  file, int line);      ⑵

在需要檢測的文件里,重定義new
#define new new(__FILE__, __LINE__)

造成的結果:
ClassName *p = new ClassName; => ClassName *p = new(__FILE__, __LINE__) ClassName;
// 實際會調用void* operator new (size_t size, const char* file, int line);

ClassName **pp = new classname[count]; => ClassName **pp = new(__FILE__, __LINE__) ClassName[count];
// 實際會調用void* operator new [] (size_t size, const char*  file, int line);

這其實是利用了placement new的語法,通過一個簡單的宏,就可以把普通的new操作對應到相應的重載( ⑴,⑵ )上去。

2、delete 的重載
void operator delete (void* p, const char* file, int line);                   ⑶
void operator delete [] (void* p, const char* file, int line);                ⑷
void operator delete (void* p);                                               ⑸
void operator delete [] (void* p);                                            ⑹

因為沒有類似于placement new的語法,所以就不能用一個宏來替換替換delete了。要調用帶有更多信息的delete操作符,只能修改源代碼了。
delete p; => delete ( p, __FILE__, __LINE__ );
delete []pp; => delete [] ( pp, __FILE__, __LINE__ );
但這個工作很煩瑣,如果并不需要多余的信息的話,簡單地重載delete( ⑸,⑹ )就可以了。

3、檢測和統計
程序開始時,在debugnew.cpp中會創建一個DebugNewTracer對象
在重載的new操作符( ⑴,⑵ )中,每一次內存分配都會被記錄,而在delete( ⑶,⑷,⑸,⑹ )中則會刪除相應的記錄。
當程序結束,DebugNewTracer對象被銷毀,它的析構函數會dump剩余的記錄,這就是泄露的內存了。

在原有代碼的基礎上,增加了記錄size的功能,這樣可以在每次new和delete時,看到實際占用的內存。所有信息可以dump出來,也可以寫入log。


5、源代碼

********************************************************
debugnew.h:

/*
 filename: debugnew.h
 
 This code is based on code retrieved from a web site. The
 author was not identified, so thanks go to anonymous.

 This is used to substitute a version of the new operator that
 can be used for debugging memory leaks. To use it:
 
 - In any (all?) code files #include debugnew.h. Make sure all
  system files (i.e. those in <>'s) are #included before
  debugnew.h, and that any header files for your own code
  are #included after debugnew.h. The reason is that some
  system files refer to ::new, and this will not compile
  if debugnew is in effect. You may still have problems
  if any of your own code refers to ::new, or if any
  of your own files #include system files that use ::new
  and which have not already been #included before
  debugnew.h.
 - Add debugnew.cpp to the CodeWarrior project or compile
  it into your Linux executable. If debugnew.cpp is in the
  project, then debugnew.h must be #included in at least
  one source file
*/

#ifndef __DEBUGNEW_H__
#define __DEBUGNEW_H__

#include <map>

#define LOG_FILE

#if defined(LOG_FILE)
#define LOG_FILE_NAME "./debugnew.log"
#endif

void* operator new (std::size_t size, const char* file, int line);
void operator delete (void* p, const char* name, int line);
void* operator new [] (std::size_t size, const char* file, int line);
void operator delete [] (void* p, const char* name, int line);

class DebugNewTracer  {
private:
 
 class Entry  {
 public:
  Entry (char const* file, int line) : _file (file), _line (line) {}
  Entry (char const* file, int line, int size) : _file (file), _line (line), _size (size) {}
  Entry () : _file (0), _line (0), _size (0) {}
  const char* File () const { return _file; }
  int Line () const { return _line; }
  size_t Size () const { return _size; }
 private:
  char const* _file;
  int _line;
  size_t _size;
 };

 class Lock {
 public:
  Lock (DebugNewTracer & DebugNewTracer) : _DebugNewTracer (DebugNewTracer) {
   _DebugNewTracer.lock ();
  }
  ~Lock () {
   _DebugNewTracer.unlock ();
  }
 private:
  DebugNewTracer & _DebugNewTracer;
 };
 typedef std::map<void*, Entry>::iterator iterator;
 friend class Lock;
 public:
 DebugNewTracer ();
 ~DebugNewTracer ();
 void Add (void* p, const char* file, int line);
 void Add (void* p, const char* file, int line, size_t size);
 void Remove (void* p);
 void Dump ();

 static bool Ready;

 private:
 void lock () { _lockCount++; }
 void unlock () { _lockCount--; }

 private:

 std::map<void*, Entry> _map;
 int _lockCount;
 size_t _totalsize;
#if defined(LOG_FILE)
 FILE* _logfp;
#endif
 };

 // The file that implements class DebugNewTracer
 // does NOT want the word "new" expanded.
 // The object DebugNewTrace is defined in that
 // implementation file but only declared in any other file.
#ifdef DEBUGNEW_CPP
DebugNewTracer DebugNewTrace;
#else
#define new new(__FILE__, __LINE__)
extern DebugNewTracer DebugNewTrace;
#endif

#endif//#ifndef __DEBUGNEW_H__


********************************************************
debugnew.cpp:

/*
 filename: debugnew.cpp

 This is used to substitute a version of the new operator that
 can be used for debugging memory leaks. In any (all?) code
 files #include debugnew.h. Add debugnew.cpp to the project.
*/

#include <iostream>
#include <map>

using namespace std;

 // This disables macro expansion of "new".
 // This statement should only appear in this file.
#define DEBUGNEW_CPP

#include "debugnew.h"

DebugNewTracer::DebugNewTracer () : _lockCount (0)
{
  // Once this object is constructed all calls to
  // new should be traced.

 Ready = true;
 _totalsize = 0;
#if defined(LOG_FILE)
 if( (_logfp=fopen(LOG_FILE_NAME,"wt")) == NULL )
 {
  printf(" Error! file: debugnew.log can not open@!\n");
  return;
 } 
 fprintf(_logfp,"new, delete list:\n");
 fflush(_logfp);
#endif
 
}
        
void DebugNewTracer::Add (void* p, const char* file, int line)
{
  // Tracing must not be done a second time
  // while it is already
  // in the middle of executing
 if (_lockCount > 0)
  return;

   // Tracing must be disabled while the map
   // is in use in case it calls new.
 DebugNewTracer::Lock lock (*this);
 _map [p] = Entry (file, line);
}

void DebugNewTracer::Add (void* p, const char* file, int line, size_t size)
{
 // Tracing must not be done a second time
 // while it is already
 // in the middle of executing
 if (_lockCount > 0)
  return;

  // Tracing must be disabled while the map
  // is in use in case it calls new.
 DebugNewTracer::Lock lock (*this);
#if 1
 //´Óȫ·¾¶ÖÐÌáÈ¡ÎļþÃû
 //linuxϵÄgcc,__FILE__½ö½ö°üÀ¨ÎļþÃû£¬windowsϵÄvc,__FILE__°üÀ¨È«Â·¾¶,ËùÒÔÓÐÕâÑùµÄ´¦Àí
 file = strrchr(file,'\\')== NULL?file:strrchr(file,'\\')+1;
#endif
 _map [p] = Entry (file, line, size);
 _totalsize += size;
#if defined(LOG_FILE)
 fprintf(_logfp,"*N p = 0x%08x, size = %6d,  %-22s, Line: %4d.  totalsize =%8d\n", p, size, file, line, _totalsize);
 fflush(_logfp);
#endif
}


void DebugNewTracer::Remove (void* p)
{
 // Tracing must not be done a second time
 // while it is already
 // in the middle of executing
 if (_lockCount > 0)
  return;

  // Tracing must be disabled while the map
  // is in use in case it calls new.
 DebugNewTracer::Lock lock (*this);

 iterator it = _map.find (p);

 if (it != _map.end ())
 {
  size_t size = (*it).second.Size();
  _totalsize -= size;
#if defined(LOG_FILE)
  fprintf(_logfp,"#D p = 0x%08x, size = %6u.%39stotalsize =%8d\n", p, size, "-----------------------------------  ", _totalsize );
  fflush(_logfp);
#endif
  _map.erase (it);
 }
 else
 {
#if defined(LOG_FILE)
  fprintf(_logfp,"#D p = 0x%08x. error point!!!\n", p );
  fflush(_logfp);
#endif
 }
}

DebugNewTracer::~DebugNewTracer ()
{
 // Trace must not be called if Dump indirectly
 // invokes new, so it must be disabled before
 // Dump is called. After this destructor executes
 // any other global objects that get destructed
 // should not do any tracing.
 Ready = false;
 Dump ();
#if defined(LOG_FILE)
 fclose(_logfp);
#endif
}

// If some global object is destructed after DebugNewTracer
// and if that object calls new, it should not trace that
// call to new.
void DebugNewTracer::Dump ()
{
 if (_map.size () != 0)
 {
  std::cout << _map.size () << " memory leaks detected\n";
#if defined(LOG_FILE)
  fprintf(_logfp, "\n\n***%d memory leaks detected\n", _map.size ());
  fflush(_logfp);
#endif
  for (iterator it = _map.begin (); it != _map.end (); ++it)
  {
   char const * file = it->second.File ();
   int line = it->second.Line ();
   int size = it->second.Size ();
   std::cout << file << ", "  << line << std::endl;
#if defined(LOG_FILE)
   fprintf(_logfp,"%s, %d, size=%d\n", file, line, size);
   fflush(_logfp);
#endif
  }
 }
 else
 {
  std::cout << "no leaks detected\n";
#if defined(LOG_FILE)
  fprintf(_logfp,"no leaks detected\n");
  fflush(_logfp);
#endif
 }

}

// If some global object is constructed before DebugNewTracer
// and if that object calls new, it should not trace that
// call to new.
bool DebugNewTracer::Ready = false;

void* operator new (size_t size, const char* file, int line)
{
 void * p = malloc (size);
 if (DebugNewTracer::Ready)
  DebugNewTrace.Add (p, file, line, size);
 return p;
}

void operator delete (void* p, const char* file, int line)
{
 file = 0; // avoid a warning about argument not used in function
 line = 0; // avoid a warning about argument not used in function
 
 if (DebugNewTracer::Ready)
  DebugNewTrace.Remove (p);
 free (p);
}
        
void* operator new [] (size_t size, const char*  file, int line)
{
 void * p = malloc (size);
 if (DebugNewTracer::Ready)
  DebugNewTrace.Add (p, file, line, size);
 return p;
}

void operator delete [] (void* p, const char* file, int line)
{
 file = 0; // avoid a warning about argument not used in function
 line = 0; // avoid a warning about argument not used in function
 
 if (DebugNewTracer::Ready)
   DebugNewTrace.Remove (p);
 free (p);
}
        
void* operator new (size_t size)
{
 void * p = malloc (size);
 // When uncommented these lines cause entries in the map for calls to new
 // that were not altered to the debugnew version. These are likely calls
 // in library functions and the presence in the dump of these entries
 // is usually misleading.
 // if (DebugNewTracer::Ready)
 //   DebugNewTrace.Add (p, "?", 0);
 return p;
}
 
void operator delete (void* p)
{
 if (DebugNewTracer::Ready)
  DebugNewTrace.Remove (p);
 free (p);
}

//add by yugang
void operator delete [] (void* p)
{
 if (DebugNewTracer::Ready)
  DebugNewTrace.Remove (p);
 free (p);
}

posted on 2008-11-02 10:52 Sandy 閱讀(756) 評論(0)  編輯 收藏 引用 所屬分類: C++
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            99伊人成综合| 亚洲一区二区视频在线| 亚洲一级电影| 亚洲美女中出| 久久久久国产精品一区二区| 国产精品视频网址| 亚洲已满18点击进入久久| 日韩一区二区精品| 久久岛国电影| 亚洲第一区中文99精品| 欧美风情在线观看| 欧美精品在线一区二区| 亚洲电影网站| 欧美高清你懂得| 欧美日韩一二三四五区| 亚洲主播在线| 亚洲欧美日韩精品久久亚洲区| 国产日韩精品一区| 久久久久国产免费免费| 亚洲欧美一区二区三区久久| 国产一区二区久久久| 久久久人成影片一区二区三区| 在线亚洲美日韩| 国产精品盗摄久久久| 欧美影视一区| 久久天天狠狠| 在线日韩中文字幕| 欧美日韩一级大片网址| 午夜精品福利电影| 久久夜色精品国产欧美乱| 亚洲精品一区二区在线观看| 欧美成人免费全部| 国产精品qvod| 久久综合久久综合久久综合| 欧美精品成人在线| 午夜精品久久| 欧美激情视频一区二区三区在线播放| 亚洲欧洲另类| 香蕉国产精品偷在线观看不卡 | 夜夜嗨av色一区二区不卡| 亚洲欧洲综合| 一区二区成人精品| 欧美日韩精品是欧美日韩精品| 免费观看成人网| 亚洲国产高清在线| 欧美国产精品久久| 美日韩精品免费观看视频| 激情六月综合| 国产精品jvid在线观看蜜臀 | 免费观看一区| 欧美一级视频| 亚洲三级影院| 国产欧美一区二区色老头| 亚洲女优在线| 亚洲毛片av在线| 久久久国产成人精品| 亚洲欧洲视频| 性娇小13――14欧美| 1024日韩| 欧美日韩一区二区三区在线| 欧美在线影院| 一区二区高清视频在线观看| 免费成人高清| 久久狠狠久久综合桃花| 亚洲一区欧美二区| 亚洲精品久久久久久久久| 精品91久久久久| 国产亚洲永久域名| 国产精品久久久久一区二区三区共 | 国产午夜精品一区二区三区视频| 欧美一级片久久久久久久| 正在播放亚洲一区| 夜夜嗨av一区二区三区四区| 日韩视频免费看| 亚洲人成网在线播放| 亚洲第一福利社区| 精品粉嫩aⅴ一区二区三区四区| 国产精品五月天| 亚洲一本大道在线| 午夜精品久久久久99热蜜桃导演| 亚洲一区免费看| 久久成人精品| 亚洲成色精品| 日韩亚洲一区二区| 亚洲欧美一区在线| 久久久亚洲高清| 欧美色精品在线视频| 国产日韩欧美视频在线| 亚洲高清一二三区| 亚洲一区国产精品| 美女视频一区免费观看| 日韩亚洲在线观看| 久久久久久一区二区| 欧美日韩国产123| 韩国av一区二区三区在线观看| 亚洲国产精品一区在线观看不卡| 亚洲欧洲精品一区二区三区| 欧美在线在线| 一本色道久久88综合亚洲精品ⅰ| 欧美在线免费看| 国产精品乱码一区二三区小蝌蚪| 国内精品久久久久国产盗摄免费观看完整版| 亚洲第一黄色网| 亚洲女人天堂成人av在线| 亚洲一区影音先锋| 91久久亚洲| 羞羞视频在线观看欧美| 最近中文字幕日韩精品| 国产精品99久久99久久久二8| 免费的成人av| 国产精品高潮呻吟视频| 欧美成人免费播放| 亚洲精品日本| 国产欧美日韩| 99精品热6080yy久久| 在线观看日韩欧美| 亚洲欧美日韩一区在线观看| 国产日韩免费| 亚洲自拍16p| 欧美在线你懂的| 91久久久亚洲精品| 99亚洲精品| 亚洲大胆人体在线| 一本色道久久综合亚洲精品小说| 欧美午夜一区二区| 美日韩精品视频免费看| 噜噜噜噜噜久久久久久91| 亚洲精品日产精品乱码不卡| 亚洲一区图片| 亚洲精品久久久久久一区二区| 亚洲一区二区av电影| 在线观看欧美成人| 亚洲午夜极品| 日韩视频专区| 久久夜色精品国产亚洲aⅴ| 一区二区三区欧美亚洲| 麻豆国产精品va在线观看不卡| 亚洲视频国产视频| 欧美成人第一页| 蜜臀va亚洲va欧美va天堂| 国产精品嫩草影院av蜜臀| 日韩视频一区二区三区在线播放| 136国产福利精品导航网址| 国产精品视频999| 99国产精品国产精品毛片| 亚洲精选久久| 欧美日韩亚洲三区| 久久亚洲私人国产精品va| 欧美亚洲色图校园春色| 亚洲国产天堂久久国产91| 欧美在线免费| 亚洲欧洲日本mm| 91久久极品少妇xxxxⅹ软件| 国产精品一区二区久久精品| 欧美伦理视频网站| 午夜激情亚洲| 国产欧美日韩一区二区三区在线观看 | 久久免费高清| 亚洲国产专区校园欧美| 欧美电影资源| 亚洲视频在线观看| 久久精品一区| 在线国产日韩| 欧美午夜不卡在线观看免费| 亚洲一区二区三区在线看| 久久成人免费| 亚洲啪啪91| 国产综合久久久久久| 奶水喷射视频一区| 亚洲在线中文字幕| 欧美成人精品激情在线观看| 亚洲影院免费观看| 亚洲福利国产| 国产欧美一区二区精品性| 久久天堂成人| 久久精品国产99| 亚洲在线视频观看| 一区二区三区不卡视频在线观看 | 在线视频欧美日韩| 狠狠色伊人亚洲综合成人| 欧美日韩午夜| 狠狠v欧美v日韩v亚洲ⅴ| 国产精品成人一区二区| 欧美片在线观看| 欧美精品一二三| 欧美视频国产精品| 国产精品99免费看 | 亚洲国语精品自产拍在线观看| 亚洲欧美日韩网| 亚洲综合国产激情另类一区| 一区二区三区久久| 中文成人激情娱乐网| 欧美风情在线观看| 欧美成人免费va影院高清| 免费久久99精品国产| 久久九九99| 欧美精品日韩综合在线| 欧美—级在线免费片| 国产精品久久久99| 国产日本欧美在线观看|