关于cvCopy函数的内容摘抄了本文的博客文章,在这里 。
一、cvcopy函数的简介
cvCopy 这个函数很熟洗哈,用得很多吧哈哈…… 以下内容来子opencv安装文件夹中自带的pdf文档。
Copies one array to another.
//复制一个数组到另外一个数组
void cvCopy(const CvArr* src, CvArr* dst, const CvArr* mask=NULL);
src The source array
//源数组,要复制谁??from whom?
//插两句题外话,opencv里面提到的数组不是通常意义上的数组,它是矩阵、图像等结构体……
dst The destination array
//目标数组,复制以后的内容给谁?? to whom?
mask
——下面这就是重点。鄙人认为很多人都没有深刻理解这个mask的作用(不要拿鞋仍我……)——
Operation mask, 8-bit single channel array; specifies elements of the destination array to
be changed
//掩码操作,mask是一个8位单通道的数组;mask指定了目标数组(dst)中那些元素是可以改变的
上面这句话还不是非常重点,重点是以下的那个公式,这个公式有多少人理解哈,就网上的资料来看很少有人理解,又或者高手都不屑于网上写东西,所以我们只能做等真相……孩子,你被mask mask住了吗 (第一个mask是opencv里面的mask数组,第二个mask是这个英文词本身的意思,不懂就google去)??真相来了!!!
The function copies selected elements from an input array to an output array:
dst(I) = src(I) if mask(I) = 0.
//该函数把输入数组(src数组)中选中的元素(可以认为是做了标记的,不过这些标志是谁来做的呢??对,就是mask,孩子你太聪明了)拷贝到…………………………………………………………………………
………………………………………………………………到哪里?快说!!拷贝到dst数组嘛……
dst(I) = src(I) if mask(I) != 0.
就是说,如果mask不是NULL,也就是说mask是一个数组,并且是一个和dst or src大小完全一致的数组。
遍历src的每一个元素,
(1)在位置i时候如果mask对应的值为不为0,那么把src (i) 的值复制给dst (i) 。
(2)如果mask(i) 为0,那么不管src(i)是什么,dst(i)都设置为0.
举例说明,这里不举太复杂的就来一个一维的就够啦。
If any of the passed arrays is of IplImage type, then its ROI and COI fields are used. Both
arrays must have the same type, the same number of dimensions, and the same size. The function
can also copy sparse arrays (mask is not supported in this case).
//如果传递给src的数组是图像类型的,那么将使用ROI或者COI。src数组和dst数组必须具有相同的数据类型、一致的数组维数、一样的大小。该函数也可以用于拷贝稀疏矩阵(但是此种情况下,mask不起作用)。
好吧,到这里了终于写完了。你现在是不是明白为什么要起mask这个名字了吧,掩码就是:不为0的时候就可一操作(具体是什么操作就看具体的函数了)、为0就掩盖住了无法操作。
补充一点 :mask = NULL 意思就是没有模板、不使用掩码操作,函数该干啥就干啥……
二、提取不规则区域的实现
由上可知,只要满足以下条件:
(1)只要mask是不规则的图像(灰度图像)
(2)mask的大小 == dst图像大小 == src图像大小
(3)在mask图像中想要提取的地方设置为全白,不想包括的地方全黑,这样效果更佳。
上代码:
// #include "cv.h" #include "cxcore.h" #include "highgui.h" int main(int argc, char* argv[]) { IplImage *pImg=cvLoadImage("lena.jpg",1); IplImage *mask=cvLoadImage("mask.jpg",0); IplImage *dstImg=cvCreateImage(cvGetSize(pImg),8,3); cvSetZero( dstImg ); cvCopy(pImg,dstImg,mask); cvNamedWindow("srcShow",1); cvNamedWindow("dstShow",1); cvNamedWindow("maskShow",1); cvShowImage("srcShow",pImg); cvShowImage("dstShow",dstImg); cvShowImage("maskShow",mask); cvWaitKey(0); cvReleaseImage(&pImg); cvReleaseImage(&mask); cvReleaseImage(&dstImage); cvDestroyWindow("srcShow"); cvDestroyWindow("dstShow"); cvDestroyWindow("maskShow"); return 0; }
有图有真相:
原图: