0
点赞
收藏
分享

微信扫一扫

(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)
  • (P54)面向对象版表达式计算器:内存泄漏跟踪器完善,表达式计算器加入内存泄漏跟踪_析构函数


  • (P54)面向对象版表达式计算器:内存泄漏跟踪器完善,表达式计算器加入内存泄漏跟踪_表达式计算_02


  • (P54)面向对象版表达式计算器:内存泄漏跟踪器完善,表达式计算器加入内存泄漏跟踪_内存泄漏_03

  • 只要涉及到加法或者乘法就会有库的泄漏
  • (P54)面向对象版表达式计算器:内存泄漏跟踪器完善,表达式计算器加入内存泄漏跟踪_内存泄漏_04


  • (P54)面向对象版表达式计算器:内存泄漏跟踪器完善,表达式计算器加入内存泄漏跟踪_析构函数_05

  • 没有+或者*就没有库的泄漏
  • (P54)面向对象版表达式计算器:内存泄漏跟踪器完善,表达式计算器加入内存泄漏跟踪_析构函数_06

  • 为什么涉及到加法或者乘法就会有库的泄漏?
    如果是+或-,就会创建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容器中,将调用链也存储起来


举报

相关推荐

线性栈实现中缀表达式计算器

0 条评论