本文主要是介绍(P54)面向对象版表达式计算器:内存泄漏跟踪器完善,表达式计算器加入内存泄漏跟踪,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
文章目录
- 1.内存泄漏跟踪器完善
- 2.表达式计算器加入内存泄漏跟踪
1.内存泄漏跟踪器完善
- eg:P54\CalculatorTest1\Tracer.h,P54\CalculatorTest1\Tracer.cpp
lockCount_跟踪器相关的代码
- 测试:
- 中间的代码很长,如下图所示,忘记调用了unlock的解决办法
可以封装一个类,在构造函数中调用lock(),在析构函数中调用unlock(),这样不论在什么地方return,局部对象总是被释放的,栈上的对象总是被释放的,它的析构函数总是被调用的,也就是说利用局部对象的析构函数的调用的确定性来进行封装
2.表达式计算器加入内存泄漏跟踪
-
将P54\CalculatorTest1\Tracer.h,P54\CalculatorTest1\Tracer.cpp,P54\CalculatorTest1\DebugNew.h放到P54\Calculator中
-
eg:
在可能产生内存泄漏的地方增加:#include "DebugNew.h"
P54\Calculator\main.cpp中增加了:status = STATUS_QUIT;//为了检查内存泄漏,就会退出程序,就会打印内存泄漏的信息
P54\Calculator\Parser.cpp增加了 //内存泄漏出现的位置
头文件包含顺序:C库,C++库,其他库,项目
-
测试:内存泄漏的地方
?是库调用的内存泄漏,调用的是operator new,void* operator new(size_t size)它没有传递file和line,而我们自己编写的程序调用new的时候会调用file和line:#define new new(FILE, LINE)
只要涉及到加法或者乘法就会有库的泄漏
没有+或者*就没有库的泄漏
-
为什么涉及到加法或者乘法就会有库的泄漏?
如果是+或-,就会创建SumNode ,class SumNode : public MultipleNode,如果是*或者/就会创建ProductNode ,class ProductNode : public MultipleNode;这两种节点的特征都是继承至MultipleNode,而MultipleNode又持有vector,vector内部也会分配空间,也是调用operate new(调用AppendChild,再调用push_back,会调用operate new),就没有传递文件名和行号,这个空间也可能存在内存泄漏。
为什么存在内存泄漏?
原因是vector对象的析构函数没有被调用,childs_是栈上对象,正常来讲应该会被释放,但是childs_它寄生于MultipleNode,如果MultipleNode节点未被释放,则它所持有的其他对象也不会被释放,即:
std::vector<Node*> childs_;std::vector<bool> positive_;//节点的正负性
这些类对象也不会被释放。后面会使用智能指针来释放这些对象,来避免内存泄漏。
- 更近一步
将栈回溯信息void Exception::FillStackTrace()也添加到map容器中,将调用链也存储起来
这篇关于(P54)面向对象版表达式计算器:内存泄漏跟踪器完善,表达式计算器加入内存泄漏跟踪的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!