opencv各种内存泄露情况的总结

举报
ShaderJoy 发表于 2021/12/30 00:46:20 2021/12/30
【摘要】 以下文章共总结出cvLoadImage cvCloneImage cvGetRow 三个函数容易由于使用不当而泄露内存,要注意! OpenCV中的内存泄露问题(cvLoadImage函数) 摘自 在做项目的过程中,使用OpenCV经常会出现一些内存泄露问题,自己编写的程序出现问题还情有可原,但若是库函数调用和使用时出...

以下文章共总结出cvLoadImage cvCloneImage cvGetRow 三个函数容易由于使用不当而泄露内存,要注意!

OpenCV中的内存泄露问题(cvLoadImage函数)

摘自

在做项目的过程中,使用OpenCV经常会出现一些内存泄露问题,自己编写的程序出现问题还情有可原,但若是库函数调用和使用时出现,却很令我恼 火。花了好长时间和实践的经验告诉我应该客服它。下面把一些检测出的问题进行化解。(可能是水平不够,这些函数使用不当,望高手指点)

cvLoadImage函数:

可能大家还觉察不出来,但我深有体会,在程序中这个函数使用一次两次感觉不来,但在处理序列图像循环调用这个函数时,内存泄露的可能让你目瞪口呆!opencv各种内存泄露情况的大总结即使你在最后使用cvReleaseImage(&pImg);进行了释放,实验证明:似乎不能成功释放。

解决方法:

使用CvvImage类代替(详见  CvvImage类参考手册)。并且使用CvvImage类的Load函数。

使用过程大概如下:


  
  1. //变量定义:
  2. CvvImage pSrcImg;
  3. IplImag *pSrcImgCopy ; //使用IplImag变量做个拷贝。毕竟IplImag 类处理方便。
  4. //获取图像:
  5. pSrcImg.Load(str); //str为Cstring类型的图像文件名
  6. pSrcImgCopy = pSrcImg.GetImage(); //拷贝出pSrcImg的图像数据。
  7. //释放内存
  8. //pSrcImg变量不需要每次释放,因为每次Load时是覆盖以前的内存区域。pSrcImgCopy 同样。
  9. //不过在程序结束时要释放,以免产生内存泄露或者别人以为你忘了。
  10. cvReleaseImage(&pSrcImgCopy );
  11. pSrcImg.Destroy();
  12. //不过要正确释放pSrcImgCopy 时,声明时必须create下:
  13. pSrcImgCopy = cvCreateImage(cvSize(IMGWIDHT,IMGHEIGHT),IPL_DEPTH_8U, 3);
  14. //IMGWIDHT,IMGHEIGHT为图像宽和高。


OpenCV中的内存泄露问题(cvCloneImage函数)

摘自

cvCloneImage函数:

这个函数也会出现内存泄露!虽然可以释放,但程序复杂不知道在那里释放,因为它每次拷贝是制作图像的完整拷贝包括头、ROI和数据。不会覆盖以前的内容。每次使用时编译器会分配内存空间。一个752*480大小的图像,每次泄露的内存大约为1M。

解决方法:

使用cvCopy函数代替。


   
  1. cvCopy(pSrcImg,pImg,NULL); //代替 pImg = cvCloneImage(pSrcImg);
  2. //pImg初始化时必须分配空间,否则上述函数不能执行。
  3. //pImg = cvCreateImage(cvSize(IMGWIDHT,IMGHEIGHT),IPL_DEPTH_8U, 3);
在使用这个函数之前,你必须用cvCreateImage()一类的函数先开一段内存,然后传递给dst。cvCopy会把src中的数据复制到dst的内存中。
cvCloneImage的原型是:
IplImage* cvCloneImage( const IplImage* image );
  

在使用函数之前,不用开辟内存。该函数会自己开一段内存,然后复制好image里面的数据,然后把这段内存中的数据返回给你。
clone是把所有的都复制过来,也就是说不论你是否设置Roi,Coi等影响copy的参数,clone都会原封不动的克隆过来。
copy就不一样,只会复制ROI区域等。

用clone复制之后,源图像在内存中消失后,复制的图像也变了,而用copy复制,源图像消失后,复制的图像不变

摘自

大概调出来两种情况造成的内存泄漏:

1.在使用clone之前没有对目标矩阵释放。比如:
 


   
  1. CvMat* a = cvCreateMat(5,5,CV_32F);
  2. CvMat* b = cvCreateMat(5,5,CV_32F);
  3. ...
  4. a = cvCloneMat(b);


这样a所指向的那个mat就会泄漏出去了。

正确的做法是在clone之前release掉原来的那个矩阵。并且注意在release之后将a置空。即:

cvReleaseMat(&a); a=NULL;
  


2.在使用cvGetRow,cvGetCols等函数时,没有对目的矩阵的数据区释放。比如:


   
  1. CvMat* a = cvCreateMat(5,5,CV_32F);
  2. CvMat* b = cvCreateMat(5,1,CV_32F);
  3. ...
  4. for(int i=0;i<a->height;i++)
  5. {
  6. cvGetRow(a,b,i);
  7. }
  8. //这样a的第i行会复制一份然后替换掉原始的b->data.这样也就造成了这一部分的内存泄漏。正确的作法是:
  9. for(..)
  10. {
  11. cvReleaseData(b);
  12. cvGetRow(a,b,i);
  13. }

 

呃,按说这些东西openCV应该能封装好了,并且这么重要的东西它也不在文档中写明。看着不多,写出的程序如果循环起来就不知道泄漏多少内存了。。。。。。

此外,再向大家推荐一个内存泄漏的检查工具:IBM Rational Purify.

==================================================================

Opencv各种内存泄露情况的总结 、Opencv各种内存泄露情况的总结2

OpenCV 中文论坛 OpenCV中的内存泄漏检测

===========================================================


文章来源: panda1234lee.blog.csdn.net,作者:panda1234lee,版权归原作者所有,如需转载,请联系作者。

原文链接:panda1234lee.blog.csdn.net/article/details/8925503

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。