关于手机内存泄露问题

发表于:2008-12-09 15:55

字体: | 上一篇 | 下一篇 | 我要投稿

 作者:未知    来源:网络转载

  一、 内存泄漏

  内存泄漏是编程中常常见到的一个问题。 内存泄漏往往会一种奇怪的方式来表现出来,基本上每个程序都表现出不同的方式。 但是一般最后的结果只有两个,一个是程序当掉。一个是系统内存不足。 还有一种就是比较介于中间的结果程序不会当,但是系统的反映时间明显降低,需要定时的Reboot才会正常。

  有一个很简单的办法来检查一个程序是否有内存泄漏。就是是用Windows的任务管理器(Task Manager)。   运行程序,然后在任务管理器里面查看 “内存使用”和”虚拟内存大小”两项,当程序请求了它所需要的内存之后,如果虚拟内存还是持续的增长的话,就说明了这个程序有内存泄漏问题。 当然如果内存泄漏的数目非常的小,用这种方法可能要过很长时间才能看的出来。

  当然最简单的办法大概就是用CompuWare的BoundChecker 之类的工具来检测了,不过这些工具的价格对于个人来讲稍微有点奢侈了。

  如果是已经发布的程序,检查是否有内存泄漏是又费时又费力。 所以内存泄漏应该在Code的生成过程就要时刻进行检查。

  二、 原因

  而内存泄漏产生的原因一般是三种情况: 1。分配完内存之后忘了回收。2。 程序Code有问题,造成没有办法回收。3。某些API函数操作不正确,造成内存泄漏。

  1、内存忘记回收,这个是不应该的事情。但是也是在代码种很常见的问题。分配内存之后,用完之后,就一定要回收。 如果不回收,那就造成了内存的泄漏,造成内存泄漏的Code如果被经常调用的话,那内存泄漏的数目就会越来越多的。从而影响整个系统的运行。 比如下面的代码

  for (int =0;I<100;I++)

  {

       Temp =   new BYTE[100];

  }

  就会产生 100*100Byte的内存泄漏。

  2、在某些时候,因为代码上写的有问题,会导致某些内存想回收都收不回来,比如下面的代码。

  Temp1 = new BYTE[100];

  Temp2 = new BYTE[100];

  Temp2 = Temp1;

  这样,Temp2的内存地址就丢掉了,而且永远都找不回了,这个时候Temp2的内存空间想回收都没有办法。

  3、API函数应用不当,在Windows提供API函数里面有一些特殊的API,比如FormatMessage。如果你给它参数中有FORMAT_MESSAGE_ALLOCATE_BUFFER,它会在函数内部New一块内存Buffer出来。但是这个 buffer需要你调用LocalFree来释放。如果你忘了,那就会产生内存泄漏。

  三、 检查方法

  一般的内存泄漏检查的确是很困难,但是也不是完全没有办法。如果你用VC的库来写东西的话,那么很幸运的是,你已经有了很多检查内存泄漏的工具,只是你想不想用的问题了。 Visual C++的Debug版本的C运行库(C Runtime Library)。它已经提供好些函数来帮助你诊断你的代码和跟踪内存泄漏。 而且最方便的地方是这些函数在Release版本中完全不起任何作用,这样就不会影响你的Release版本程序的运行效率。

  比如下面的例子里面,有一个明细的内存泄漏。当然如果只有这么几行代码的话,是很容易看出有内存泄漏的。但是想在成千上万行代码里面检查内存泄漏问题就不是那么容易了。

       char * pstr = new char[5];

       lstrcpy(pstr,"Memory leak");

  我们如果我们在Debug版本的Code里面对堆(Heap)进行了操作,包括malloc, free, calloc, realloc, new, 和 delete可以利用VC Debug运行时库中堆Debug函数来做堆的完整性和安全性检查。 比如上面的代码,lstrcpy的操作明显破坏了pstr的堆结构。使其溢出,并破坏了临近的数据。那我们可以在调用lstrcpy之后的代码里面加入 _CrtCheckMemory函数。_CrtCheckMemory函数发现前面的lstrcpy使得pstr的堆结构被破坏,会输出这样的报告:

  emory check error at 0x00372FA5 = 0x79, should be 0xFD.

  memory check error at 0x00372FA6 = 0x20, should be 0xFD.

  memory check error at 0x00372FA7 = 0x6C, should be 0xFD.

  memory check error at 0x00372FA8 = 0x65, should be 0xFD.

  DAMAGE: after Normal block (#41) at 0x00372FA0.

  Normal located at 0x00372FA0 is 5 bytes long.

  它告诉说 pstr的长度应该时5个Bytes,但是在5Bytes后面的几个Bytes也被非法改写了。提醒你产生了越界操作。

  _CrtCheckMemory的返回值只有TRUE和FALSE,那么你可以用_ASSERTE()来报告出错信息。 上面的语句可以换成 _ASSERTE(_CrtCheckMemory()); 这样Debug版本的程序在运行的时候就会弹出一个警告对话框,如图1,这样就不用在运行时候一直盯着Output窗口看了。这个时候按Retry,就可以进入源代码调试了。看看问题到底出在哪里

  其他类似的函数还有_CrtDbgReport, _CrtDoForAllClientObjects, _CrtDumpMemoryLeaks,_CrtIsValidHeapPointer, _CrtIsMemoryBlock, _CrtIsValidPointer,_CrtMemCheckpoint, _CrtMemDifference, _CrtMemDumpAllObjectsSince, _CrtMemDumpStatistics, _CrtSetAllocHook, _CrtSetBreakAlloc, _CrtSetDbgFlag,_CrtSetDumpClient, _CrtSetReportFile, _CrtSetReportHook, _CrtSetReportMode

  这些函数全部都可以用来在Debug版本中检查内存的使用情况。具体怎么使用这些函数就不在这里说明了,各位可以去查查MSDN。

21/212>
《2023软件测试行业现状调查报告》独家发布~

精彩评论

  • define_NULL
    2008-12-10 13:06:32

    呵呵,顶!
    这个必须在编码的时候就要注意的问题吧。只是针对vc的,但是对于其他开发平台的就难办了

关注51Testing

联系我们

快捷面板 站点地图 联系我们 广告服务 关于我们 站长统计 发展历程

法律顾问:上海兰迪律师事务所 项棋律师
版权所有 上海博为峰软件技术股份有限公司 Copyright©51testing.com 2003-2024
投诉及意见反馈:webmaster@51testing.com; 业务联系:service@51testing.com 021-64471599-8017

沪ICP备05003035号

沪公网安备 31010102002173号