OpenCV中的三种图像数据结构CvMat, IplImage和Mat(二)IplImage
本文为原创,若有错误欢迎批评指正!
一. IplImage结构体构成
IplImage比CvMat要复杂一些,结构体组成如下:
typedef struct _IplImage { int nSize; /* IplImage大小 */ int ID; /* 版本 (=0)*/ int nChannels; /* 大多数OPENCV函数支持1,2,3 或 4 个通道 */ int alphaChannel; /* 被OpenCV忽略 */ int depth; /* 像素的位深度: IPL_DEPTH_8U, IPL_DEPTH_8S, IPL_DEPTH_16U, IPL_DEPTH_16S, IPL_DEPTH_32S, IPL_DEPTH_32F and IPL_DEPTH_64F 可支持 */ char colorModel[4]; /* 被OpenCV忽略 */ char channelSeq[4]; /* 被OpenCV忽略 */ int dataOrder; /* 0 - 交叉存取颜色通道, 意思是在每一行的结构都是BGRBGRBGR... 1 - 分开的颜色通道,把几个颜色通道分为几个颜色平面存储 cvCreateImage只能创建交叉存取图像 */ int origin; /* 0 - 顶—左结构,1 - 底—左结构 (Windows bitmaps 风格) */ int align; /* 图像行排列 (4 or 8). OpenCV 忽略它,使用 widthStep 代替 */ int width; /* 图像宽像素数 */ int height; /* 图像高像素数*/ struct _IplROI *roi; /* 图像感兴趣区域. 当该值非空只对该区域进行处理 */ struct _IplImage *maskROI; /* 在 OpenCV中必须置NULL */ void *imageId; /* 同上*/ struct _IplTileInfo *tileInfo; /*同上*/ int imageSize; /* 图像数据大小(在交叉存取格式下imageSize=image->height*image->widthStep),单位字节*/ char *imageData; /* 指向排列的图像数据 */ int widthStep; /* 排列的图像行大小,以字节为单位 */ int BorderMode[4]; /* 边际结束模式, 被OpenCV忽略 */ int BorderConst[4]; /* 同上 */ char *imageDataOrigin; /* 指针指向一个不同的图像数据结构(不是必须排列的),是为了纠正图像内存分配准备的 */ } IplImage;
这里比较重要的参数是height、width、depth和nChannels。
二. 访问图像数据
与CvMat类似,都有三种方法。
首先是函数的方法:
/*函数的方法*/ IplImage* img=cvLoadImage(filename); CvScalar s; /*sizeof(s) == img->nChannels*/ s=cvGet2D(img,i,j); /*get the (i,j) pixel value*/ cvSet2D(img,i,j,s); /*set the (i,j) pixel value*/
采用宏的方法:
/*宏*/ IplImage* img; //malloc memory by cvLoadImage or cvCreateImage for(int row = 0; row < img->height; row++) { for (int col = 0; col < img->width; col++) { b = CV_IMAGE_ELEM(img, UCHAR, row, col * img->nChannels + 0); g = CV_IMAGE_ELEM(img, UCHAR, row, col * img->nChannels + 1); r = CV_IMAGE_ELEM(img, UCHAR, row, col * img->nChannels + 2); } }
接下来是最推崇的使用指针的方法。在这之前要弄清楚IplImage与Cvmat的一些差别,剩下的用法类似。IplImage在计算行地址的时候也要采用行长度widthStep而不是图像的宽度,与CvMat中step类似。IplImage图像的首地址记作imageData,与CvMat中data类似。IplImage访问时指针设置为(uchar*),而CvMat中data为联合类型,必须说明使用的指针类型。
/*指针访问*/ IplImage* img; //malloc memory by cvLoadImage or cvCreateImage uchar b, g, r; // 3 channels for(int row = 0; row < img->height; row++) { for (int col = 0; col < img->width; col++) { b = ((uchar *)(img->imageData + row * img->widthStep))[col * img->nChannels + 0]; g = ((uchar *)(img->imageData + row * img->widthStep))[col * img->nChannels + 1]; r = ((uchar *)(img->imageData + row * img->widthStep))[col * img->nChannels + 2]; } }
三. 一个比较综合的应用
这里定义一个稍微复杂一点的应用,包括了对图像的常用操作,基本上足够用来实现其他的图像算法了。
首先读入一幅图像,若为3通道的则转化为一通道,然后将图像右下角1/4置为白色。
#include <opencv2\opencv.hpp> int main() { IplImage *img = cvLoadImage("D:\\_Gonzalez\\ch02\\Fig0205(a)(cktboard_200dpi).tif"); //获取图像的参数 printf("width=%d\n", img->width); printf("height=%d\n", img->height); printf("channel=%d\n", img->nChannels); if (img->nChannels == 3) { //新建一个与img一样大的一通道空图像 IplImage *dst = cvCreateImage(cvSize(img->width, img->height), img->depth, 1); //将彩色图像转化为一通道灰度图 cvCvtColor(img, dst, CV_RGB2GRAY); img = dst; } for (int y = img->height / 2; y < img->height; y++) { uchar* ptr = (uchar*)(img->imageData + y*img->widthStep); for (int x = img->width / 2; x < img->width; x++) { ptr[x + 1] = 255; ptr[x + 2] = 255; ptr[x + 3] = 255; } } cvNamedWindow("example", CV_WINDOW_AUTOSIZE); cvShowImage("example", img); cvWaitKey(0); cvReleaseImage(&img); cvDestroyWindow("example"); }
如果将要求改为将三通道的img右下角1/4置为白色,相应操作变成:
for (int y = img->height / 2; y < img->height; y++) { uchar* ptr = (uchar*)(img->imageData + y*img->widthStep); for (int x = img->width / 2; x < img->width; x++) { ptr[x + 1] = 255; ptr[x + 2] = 255; ptr[x + 3] = 255; /*也可以是 *(ptr+3*x+1) = 255; *(ptr+3*x+2) = 255; *(ptr+3*x+2) = 255; */ } }