OpenCV学习之子图像操作

我们有时候希望对图像的某个小部分进行操作,而不是对整个图像进行运算。有两种方法,ROI和widthStep。

1. ROI

设置一个ROI区域使用cvSetImageROI,取消一个ROI使用cvResetImageROI。函数原型分别如下:

void cvSetImageROI( IplImage* image, CvRect rect );

void cvResetImageROI( IplImage* image );

rect就是我们要操作的图像的某个区域。实例如下:


#include <cv.h>
#include <highgui.h>


int main(int argc, char* argv[])
{

IplImage* src;

CvRect rect = cvRect(70, 70, 50, 50); 

if (argc == 2 && ((src = cvLoadImage(argv[1], 1)) != NULL)) {

cvSetImageROI(src, rect);

cvAddS(src, cvScalar(50), src);

cvResetImageROI(src);

cvNamedWindow("roi_add", CV_WINDOW_AUTOSIZE);

cvShowImage("roi_add", src);

cvWaitKey();

cvReleaseImage(&src);

cvDestroyWindow("roi_add");

}

return 0;

}

这里我们将ROI区域的灰度值增加50。


2. widthStep

有了ROI,为什么还需要使用widthStep。ROI只能串行的处理,需要不断设置和重置,而使用widthStep可以连续的操作多个子区域。实例如下:

#include <cv.h>
#include <highgui.h>


int main(int argc, char* argv[])
{

IplImage* src;

CvRect rect = cvRect(100, 20, 50, 50);

if (argc == 2 && ((src = cvLoadImage(argv[1], 1)) != NULL)) {

IplImage* sub_img = cvCreateImageHeader(cvSize(rect.width, rect.height),

src->depth, src->nChannels);

sub_img->origin = src->origin;

sub_img->widthStep = src->widthStep;


sub_img->imageData = src->imageData +

rect.y * src->widthStep +

rect.x * src->nChannels;


cvAddS(sub_img, cvScalar(50), sub_img);

cvNamedWindow("sub_add", CV_WINDOW_AUTOSIZE);

cvShowImage("sub_add", src);

cvWaitKey();

   

cvReleaseImageHeader(&sub_img);

cvDestroyWindow("sub_add");

cvReleaseImage(&src);


}

return 0;
}

注意:

在这里我们只创建了一个子图像的头,它的数据域在原图像中。我们通过widthStep来操作原图像的数据,从而达到修改原图像子区域的操作。

子图像的widhtStep等于原图像的widthStep,因为操作了一行后,欲操作下一行,必须将数据域指针加上widthStep。

posted @ 2012-04-13 11:14  移动应用开发  阅读(716)  评论(0编辑  收藏  举报