opencv各种内存泄露情况的总结
以下文章共总结出cvLoadImage cvCloneImage cvGetRow 三个函数容易由于使用不当而泄露内存,要注意!
OpenCV中的内存泄露问题(cvLoadImage函数)
摘自在做项目的过程中,使用OpenCV经常会出现一些内存泄露问题,自己编写的程序出现问题还情有可原,但若是库函数调用和使用时出现,却很令我恼 火。花了好长时间和实践的经验告诉我应该客服它。下面把一些检测出的问题进行化解。(可能是水平不够,这些函数使用不当,望高手指点)
cvLoadImage函数:
可能大家还觉察不出来,但我深有体会,在程序中这个函数使用一次两次感觉不来,但在处理序列图像循环调用这个函数时,内存泄露的可能让你目瞪口呆!即使你在最后使用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复制,源图像消失后,复制的图像不变
大概调出来两种情况造成的内存泄漏:
-
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、
===========================================================
文章来源: panda1234lee.blog.csdn.net,作者:panda1234lee,版权归原作者所有,如需转载,请联系作者。
原文链接:panda1234lee.blog.csdn.net/article/details/8925503
- 点赞
- 收藏
- 关注作者
评论(0)