一:背景

上一篇我们聊到了如何去找 热点函数,这一篇我们来看下当你的程序出现了 非托管内存泄漏 时如何去寻找可疑的代码源头,其实思路很简单,就是在 HeapAlloc 或者 VirtualAlloc 时做 Hook 拦截,记录它的调用栈以及分配的内存量, PerfView 会将这个 分配量 做成一个 权重,最后可以根据 权重 高低来找到有问题的调用栈。

二:案例演示

为了方便讲述,我们演示一个 Windows 的 Nt堆 内存泄漏,让 C# 调用 C++ 代码时故意泄漏内存,代码如下:


#include <iostream> extern "C"
{
_declspec(dllexport) int calc_size(int size);
} int calc_size(int size) { int* buffer = new int[size]; return 2 * size;
}

然后在 C# 中导入这个 C++ 的 dll。


internal class Program
{
[DllImport("ConsoleApplication2.dll", CallingConvention = CallingConvention.Cdecl)]
extern static int calc_size(int size); static void Main(string[] args)
{
for (int i = 0; i < int.MaxValue; i++)
{
var size = calc_size(1000); Console.WriteLine($"i={i}");
}
}
}

接下来把程序跑起来,再打开 Perfview,在 OS Heap Process 输入框填入进程号19404,监控 15s然后 Start Collection即可,这么做的目的时拦截 HeapAllocHeapFree 方法,截图如下:

稍等 15s 之后,打开 Memory / Net OS Heap Alloc Stacks 选项卡。

接下来切到 CallTree 选项卡,清除掉 GroupPats 中的条件,观察各自的调用栈,截图如下:

从图中的 inc % 列可以看到,calc_size 方法的 分配权重量 占 总分配量的 99.9% ,这就说明此方法有很大的嫌疑,最后就是查看源码了哦。

最新文章

  1. Android实现Layout缩放动画
  2. Unity使用Windows弹窗保存图片
  3. SQL 报错信息整理及解决方案(持续更新)
  4. tomcat PermGen space
  5. C#编写记事本(高仿)
  6. HDU 1394-Minimum Inversion Number(BIT)
  7. (hdu)1950 Bridging signals(最长上升子序列)
  8. BZOJ 1018 堵塞的交通
  9. python脚本0b文件处理
  10. Mybash的实现
  11. SASS 中变量的默认值
  12. 移动APP及游戏推广,有预算为什么还起不了量
  13. 我的 FPGA 学习历程(06)—— 二进制转格雷码
  14. 全面超越Appium,使用Airtest超快速开发App爬虫
  15. 综合评价模型C++实现
  16. git-创建新项目
  17. dapper.net 转载
  18. 【NOIP2018】保卫王国 动态dp
  19. cesium编程入门(八)设置材质
  20. [Python] dict字典的浅复制与深复制

热门文章

  1. 『忘了再学』Shell基础 — 19、使用declare命令声明变量类型
  2. 关于我学git这档子事(5)
  3. TornadoFx设置保存功能((config和preference使用))
  4. 第06组 Beta冲刺 (4/5)
  5. Lifted ElGamal 门限加密算法
  6. systemctl设置程序开机启动、关闭、启用/禁用服务以vsftpd为例
  7. 覆盖率检查工具:JaCoCo 食用指南
  8. 【转载】k8s入坑之路(2)kubernetes架构详解
  9. 谷歌浏览器Chrome官方下载地址
  10. 【Spring】AOP实现原理(一):AOP基础知识