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

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

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

摘自

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

cvLoadImage函数:

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

解决方法:

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

使用过程大概如下:

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


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

摘自

cvCloneImage函数:

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

解决方法:

使用cvCopy函数代替。

cvCopy(pSrcImg,pImg,NULL);   //代替 pImg = cvCloneImage(pSrcImg);
//pImg初始化时必须分配空间,否则上述函数不能执行。
//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之前没有对目标矩阵释放。比如:
 

CvMat* a = cvCreateMat(5,5,CV_32F);
CvMat* b = cvCreateMat(5,5,CV_32F);
...
a = cvCloneMat(b);


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

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

cvReleaseMat(&a); a=NULL;


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

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

 

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

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

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

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

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

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


  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值