PerfView专题 (第二篇):如何寻找 C# 中的 Heap堆内存泄漏 ...

打印 上一主题 下一主题

主题 867|帖子 867|积分 2601

一:背景

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

为了方便讲述,我们演示一个 Windows 的 Nt堆 内存泄漏,让 C# 调用 C++ 代码时故意泄漏内存,代码如下:
  1. #include <iostream>
  2. extern "C"
  3. {
  4.         _declspec(dllexport) int calc_size(int size);
  5. }
  6. int calc_size(int size) {
  7.         int* buffer = new int[size];
  8.         return 2 * size;
  9. }
复制代码
然后在 C# 中导入这个 C++ 的 dll。
  1.     internal class Program
  2.     {
  3.         [DllImport("ConsoleApplication2.dll", CallingConvention = CallingConvention.Cdecl)]
  4.         extern static int calc_size(int size);
  5.         static void Main(string[] args)
  6.         {
  7.             for (int i = 0; i < int.MaxValue; i++)
  8.             {
  9.                 var size = calc_size(1000);
  10.                 Console.WriteLine($"i={i}");
  11.             }
  12.         }
  13.     }
复制代码
接下来把程序跑起来,再打开 Perfview,在 OS Heap Process 输入框填入进程号19404,监控 15s然后 Start Collection即可,这么做的目的时拦截 HeapAlloc 和 HeapFree 方法,截图如下:

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

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

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

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

悠扬随风

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表