先来说下实现思路:可以实现一个Trace类,调用 operator new 的时候就将指向分配内存的指针、当前文件、当前行等信息添加进Trace 成员map容器内,在调用operator delete 的时候删除这些信息。定义一个全局Trace 对象,当程序结束,对象析构时判断成员map 是否还有信息,如果有则打印出来,表示已经发生内存泄漏,从输出可以看出是哪一个文件哪一行分配了内存但没有释放掉。

DebugNew.h:

 C++ Code 

1

2

3

4

5

6

7

8

9

 
#ifndef _DEBUG_NEW_H_


#define _DEBUG_NEW_H_

#ifndef NDEBUG


#include 
"Tracer.h"


#define 
new 
new(__FILE__, __LINE__)


#endif 
// NDEBUG

#endif 
// _DEBUG_NEW_H_

Trace.h:

 C++ Code 

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

 
#ifndef _TRACER_H_


#define _TRACER_H_

#include <map>

#ifndef NDEBUG

void *
operator 
new(size_t size, 
const 
char *file, 
long line);


void 
operator 
delete(
void *p);

void *
operator 
new[](size_t size, 
const 
char *file, 
long line);


void 
operator 
delete[](
void *p);

class Tracer

{


private:

    
class Entry

    {

    
public:

        Entry(
const 
char *file = 

long line = 
)

            : file_(file), line_(line) {}

        
const 
char *File() 
const

        {

            
return file_;

        }

        
long Line() 
const

        {

            
return line_;

        }

    
private:

        
const 
char *file_;

        
long line_;

    };


public:

    Tracer();

    ~Tracer();

    
static 
bool Ready;

void Add(
void *p, 
const 
char *file, 
long line);

    
void Remove(
void *p);

    
void Dump();

private:

    std::map<
void *, Entry> mapEntry_;

};

#endif 
// NDEBUG

#endif 
// _TRACER_H_

Trace.cpp:

 C++ Code 

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77

78

79

80

81

82

83

84

85

86

87

88

89

90

91

92

93

94

 
#include <iostream>


#include 
"Tracer.h"

#ifndef NDEBUG

bool Tracer::Ready = 
false;

Tracer::Tracer()

{

    Ready = 
true;

}

Tracer::~Tracer()

{

    Ready = 
false;

    Dump();

}

void Tracer::Add(
void *p, 
const 
char *file, 
long line)

{

    mapEntry_[p] = Entry(file, line);

}

void Tracer::Remove(
void *p)

{

    std::map<
void *, Entry>::iterator it;

    it = mapEntry_.find(p);

    
if (it != mapEntry_.end())

    {

        mapEntry_.erase(it);

    }

}

void Tracer::Dump()

{

    
if (mapEntry_.size() > 
)

    {

        std::cout << 
"*** Memory leak(s):" << std::endl;

        std::map<
void *, Entry>::iterator it;

for (it = mapEntry_.begin(); it != mapEntry_.end(); ++it)

        {

            
const 
char *file = it->second.File();

            
long line = it->second.Line();

            
int addr = 
reinterpret_cast<
int>(it->first);

            std::cout << 
"0x" << std::hex << addr << 
": "

                      << file << 
", line " << std::dec << line << std::endl;

}

        std::cout << std::endl;

    }

}

Tracer NewTrace;

void *
operator 
new(size_t size, 
const 
char *file, 
long line)

{

    
void *p = malloc(size);

    
if (Tracer::Ready)

    {

        NewTrace.Add(p, file, line);

    }

    
return p;

}

void 
operator 
delete(
void *p)

{

    
if (Tracer::Ready)

    {

        NewTrace.Remove(p);

    }

    free(p);

}

void *
operator 
new[](size_t size, 
const 
char *file, 
long line)

{

    
void *p = malloc(size);

    
if (Tracer::Ready)

    {

        NewTrace.Add(p, file, line);

    }

    
return p;

}

void 
operator 
delete[](
void *p)

{

    
if (Tracer::Ready)

    {

        NewTrace.Remove(p);

    }

    free(p);

}


#endif 
// #ifndef NDEBUG

main.cpp:

 C++ Code 

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

 
#include <iostream>


using 
namespace std;

#include 
"DebugNew.h"

int main(
void)

{

    
int *p = 
new 
int;

    
/*delete p;*/

int *p2 = 
new 
int[
];

    
/*delete[] p2;*/

return 
;

}

程序 #define new new(__FILE__, __LINE__); 是为了利用__FILE__, 和 __LINE__两个宏,分别代表文件名和行数。分别重载了

operator new 和 operator new[]  函数以及对应的delete,更详细的讨论可以参见这里。当全局对象NewTrace 析构时调用Dump成员

函数,如果new 和 delete 没有匹配,那么map将存在泄漏信息,并打印出来。

此外只在Debug版本(没有定义NDEBUG)才跟踪内存泄漏,所以加上#ifndef NDEBUG ... #endif

而由于一般的C++库中可能没有#define new new(__FILE__, __LINE__);  即调用的还是原始的new,但现在程序中并没有重载这种类

型的new和delete函数,故并不能跟踪类似map容器之类的内存泄漏,但一般正常使用C++库容器的话,是不会造成内存泄漏的,

C++库已经实现得比较完善了,至少比我们自己写的程序要好很多。

参考:

C++ primer 第四版
Effective C++ 3rd
C++编程规范

从零开始学C++之重载 operator new 和 operator delete 实现一个简单内存泄漏跟踪器的更多相关文章

  1. operator new 和 operator delete 实现一个简单内存泄漏跟踪器

    先来说下实现思路:可以实现一个Trace类,调用 operator new 的时候就将指向分配内存的指针.当前文件.当前行等信息添加进Trace 成员map容器内,在调用operator delete ...

  2. 从零开始学JAVA(03)-用Eclipse生成HelloWorld的Jar文件(简单不带包)

    前面已经编写了helloWorld的程序,也可以在Eclipse IDE中正常运行,但如何脱离IDE运行呢? 先通过代码生成JAR文件,选择“File→Export...”,弹出Export对话框,选 ...

  3. 尝试从零开始构建我的商城 (一) :使用Abp vNext快速一个简单的商城项目

    尝试从零开始构建我的商城 (一) :使用Abp vNext快速搭建一个简单的项目 前言 GitHub地址 https://github.com/yingpanwang/MyShop 此文目的 本文将尝 ...

  4. 函数定义从零开始学C++之从C到C++(一):const与#define、结构体对齐、函数重载name mangling、new/delete 等

    今天一直在学习函数定义之类的问题,下午正好有机会和大家共享一下. 一.bool 类型 逻辑型也称布尔型,其取值为true(逻辑真)和false(逻辑假),存储字节数在不同编译系统中可能有所不同,VC+ ...

  5. 从零开始学C++之运算符重载(三):完善String类([]、 +、 += 运算符重载)、>>和<<运算符重载

    在前面文章中使用过几次String类的例子,现在多重载几个运算符,更加完善一下,并且重载流类运算符. []运算符重载 +运算符重载 +=运算符重载 <<运算符重载 >>运算符重 ...

  6. 从零开始学 Web 之 JavaScript(三)函数

    大家好,这里是「 Daotin的梦呓 」从零开始学 Web 系列教程.此文首发于「 Daotin的梦呓 」公众号,欢迎大家订阅关注.在这里我会从 Web 前端零基础开始,一步步学习 Web 相关的知识 ...

  7. 从零开始学 Web 之 Ajax(三)Ajax 概述,快速上手

    大家好,这里是「 从零开始学 Web 系列教程 」,并在下列地址同步更新...... github:https://github.com/Daotin/Web 微信公众号:Web前端之巅 博客园:ht ...

  8. 从零开始学Kotlin-扩展函数(10)

    从零开始学Kotlin基础篇系列文章 什么是扩展函数 扩展函数数是指在一个类上增加一种新的行为,我们甚至没有这个类代码的访问权限: Kotlin 可以对一个类的属性和方法进行扩展,且不需要继承或使用 ...

  9. 关东升的《从零开始学Swift》3月9日已经上架

    大家一直期盼的<从零开始学Swift>于3月9日已经上架,它是关东升老师历时8个月的呕心沥血所编著,全书600多页,此本书基于Swift 2.x,通过大量案例全面介绍苹果平台的应用开发.全 ...

随机推荐

  1. CentOs Linux 常见命令

    整理一些常用的命令(持续更新): 查看端口是否开启: netstat -an | grep prot (查看是否打开23端口) |:通道的意思,grep是指查看当前字符所在的行 LINUX通过下面的命 ...

  2. MAC中在eclipse luna上搭建移动平台自动化测试框架(UIAutomator/Appium/Robotium/MonkeyRunner)关键点记录

    这几天因为原来在用的hp laptop的电池坏掉了,机器一不小心就断电,所以只能花时间在自己的macbook pro上重新搭建整套环境,大家都知道搭建环境是个很琐碎需要耐心的事情,特别是当你搭建的安卓 ...

  3. Model和Entity Framework

    Model和Entity Framework 上一节:ASP.NET MVC 5 入门教程 (4) View和ViewBag 下一节:ASP.NET MVC5 + EF6 入门教程 (6) View中 ...

  4. Java 多并发之原子访问(Atomic Access)

    在编程中,一个原子操作是只会出现一次的.一个原子操作在中间不会停止:要么全部发生要么一点也不发生.我们只有在原子操作完成之后才会看到原子操作的具体影响. 甚至是非常简单的表达式能够构造分解为简单操作的 ...

  5. microsoft NLayerApp项目中的层次结构图

    microsoft NLayerApp项目中的层次结构图 回到目录 如果你想学好一样东西,一定要看高手是如何做的 如果你想学好.net,一定要看.net framworks源代码 如果你想学好分层结构 ...

  6. Eclipse在SVN安装步骤(两种)和使用方法

    一.至Eclipse安装SVN,最常见的两种方式:手动模式,使用安装向导.具体操作步骤如下: 单程:手动安装 1.官方网站下载,从site-1.6.9.zip文件,网址是:subclipse.tigr ...

  7. 使用JasperReport+iReport进行Web报表开发

    使用JasperReport+iReport进行Web报表开发 前言 在实际工程中非常,报告是其中很重要的一部分,结果以报表的形式呈现出来.这里所提到的报表可不是简单的二维表,而是拥有复杂表头的.多维 ...

  8. TodoList开发笔记 – PartⅠ

    做了一年多的桌面软件,最近开始转向Web方面的开发,既然比较熟悉Net那么首当其冲就是学习ASP.Net,以及HTML.CSS.Javascript. 为了检验这个把星期来的学习成果,着手做了一个To ...

  9. ASP.NET MVC之单元测试

    ASP.NET MVC之单元测试分分钟的事2014-07-15 13:05 by 书洞里的猫, 550 阅读, 4 评论, 收藏, 编辑 一.为什么要进行单元测试? 大部分开发者都有个习惯(包括本人在 ...

  10. SQL Server 增删改

    --use用来设置当前使用哪个数据库use StudentDb--go批处理go --T-SQL中不区分大小写,数据库表中的数据是区分大小写的--例如:insert与INSERT不区分大小写,数据库表 ...