关于IplImage中widthstep的大小与width,nchannels等的关系的问题_widthstep

http://www.niubb.net/a/2015/12-02/1104644.html

话题:widthstep

width是图像宽度,可为任意值;widthstep是行字节数,应该是4的倍数,不一定等于width,nchannels为图像通道数。
#define WIDTHBYTES(bits) (((bits)+31)/32*4) 看看这个公式你就明白了。

对ROI和widthStep的补充

ROI和widthStep在实际工作中有很重要的作用,在很多情况下,使用它们会提高计算机视觉代码的执行速度。这是因为它们允许对图像的某一小部分进行操作,而不是对整个图像进行运算。在OpenCV中,普遍支持ROI和widthStep,函数的操作被限于感兴趣区域。要设置或取消ROI,就要使用cvSetImageROI()和cvResetImageROI()函数。如果想设置ROI,可以使用函数cvSetImageROI(),并为其传递1个图像指针和矩形。而取消ROI,只需要为函数cvResetImageROI()传递1个图像指针。

  1. voidcvSetImageROI( IplImage* image, CvRect rect);
  2. void cvResetImageROI( IplImage*image );

为了解释ROI的用法,我们假设要加载一幅图像并修改一些区域,如例3-12的代码,读取了一幅图像,并设置了想要的ROI的x,y,width和height的值,最后将ROI区域中像素都加上1个整数。本例程中通过内联的cvRect()构造函数设置ROI。通过cvResetImageROI()函数释放ROI是非常重要的,否则,将忠实地只显示ROI区域。

例3-12:用imageROI来增加某范围的像素

  1. //roi_add <image><x><y><width><height><add>
  2. #include <cv.h>
  3. #include<highgui.h>
  4. int main(int argc,char** argv)
  5. {
  6. IplImage* src;
  7. if( argc == 7&& ((src=cvLoadImage(argv[1],1)) != 0))
  8. {
  9. int x =atoi(argv[2]);
  10. int y =atoi(argv[3]);
  11. int width =atoi(argv[4]);
  12. int height =atoi(argv[5]);
  13. int add =atoi(argv[6]);
  14. cvSetImage ROI(src, cvRect(x,y,width,height));
  15. cvAddS(src, cvScalar(add),src);
  16. cvResetImageROI(src);
  17. cvNamedWindow( "Roi_Add", 1 );
  18. cvShowImage( "Roi_Add", src );
  19. cvWaitKey();
  20. }
  21. return 0;
  22. }

使用例3-12中的代码把ROI集中于一张猫的脸部,并将其蓝色通道增加150后的效果如图3-3所示。【45~46】

关于IplImage中widthstep的大小与width,nchannels等的关系的问题
(点击查看大图)图3-3:在猫脸上用ROI增加150像素的效果

通过巧妙地使用widthStep,我们可以达到同样的效果。要做到这一点,我们创建另1个图像头,让它的width和height的值等于interest_rect的width和height的值。我们还需要按interest_rect起点设置图像起点(左上角或者左下角)。下1步,我们设置子图像的widthStep与较大的interest_img相同。这样,就可以在子图像中逐行地步进到大图像里子区域中下一行开始处的合适位置。最后设置子图像的imageDate指针指向兴趣子区域的开始,如例3-13所示。

例3-13:利用其他widthStep方法把interest_img的所有像素值增加1

  1. // AssumingIplImage *interest_img; and
  2. // CvRectinterest_rect;
  3. // Use widthStepto get a region of interest
  4. //
  5. // (Alternatemethod)
  6. //
  7. IplImage*sub_img =cvCreateImageHeader(
  8. cvSize(
  9. interest_rect.width,
  10. interest_rect.height
  11. ),
  12. interest_img->depth,
  13. interest_img->nChannels
  14. );
  15. sub_img->origin= interest_img->origin;
  16. sub_img->widthStep = interest_img->widthStep;
  17. sub_img->imageData = interest_img->imageData +
  18. interest_rect.y * interest_img->widthStep+
  19. interest_rect.x * interest_img->nChannels;
  20. cvAddS( sub_img,cvScalar(1), sub_img );
  21. cvReleaseImageHeader(&sub_img);

看起来设置和重置ROI更方便一些,为什么还要使用widthStep?原因在于有些时候在处理的过程中,想在操作过程中设置和保持一幅图像的多个子区域处于活动状态,但是ROI只能串行处理并且必须不断地设置和重置。

最后,我们要在此提到1个词--掩码或模板,在代码示例中cvAddS()函数允许第4个参数默认值为空:const CvArr*mask=NULL。这是1个8位单通道数组,它允许把操作限制到任意形状的非0像素的掩码区,如果ROI随着掩码或模板变化,进程将会被限制在ROI和掩码的交集区域。掩码或模板只能在指定了其图像的函数中使用。

分享:黑翼天使 > 关于IplImage中widthstep的大小与width,nchannels等的关系的问题

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
#include "cv.h" #include "highgui.h" #include "cxcore.h" /*--------------copyright-hanshanbuleng--------------------*/ // 将色调H的取值范围转换到0~180之间 int main() { float H,S,V,H1,S1,V1; IplImage *src = cvLoadImage("F:\\vs2010program\\RGB_HSV\\study_test\\2.jpg", 1); IplImage *hsv_img = cvCreateImage(cvGetSize(src), 8 , 3); IplImage *h_img = cvCreateImage(cvGetSize(src), 8, 1); IplImage *s_img = cvCreateImage(cvGetSize(src), 8, 1); IplImage *v_img = cvCreateImage(cvGetSize(src), 8, 1); cvCvtColor(src, hsv_img, CV_BGR2HSV); cvSplit(hsv_img, h_img, s_img, v_img, NULL); for(int y = 0; y < hsv_img->height; y++){ for(int x = 0; x < hsv_img->width; x++) { H1 = cvGetReal2D(h_img, y, x); S1 = cvGetReal2D(s_img, y, x); V1 = cvGetReal2D(v_img, y, x); //地址法 H = (uchar)h_img->imageData[y*h_img->widthStep + x*h_img->nChannels]; S = (uchar)s_img->imageData[y*s_img->widthStep + x*s_img->nChannels]; V = (uchar)v_img->imageData[y*v_img->widthStep + x*v_img->nChannels]; printf("H:%f S:%f V:%f \n",H,S,V); } } cvNamedWindow("hsv_img", 0); //HSV图 cvShowImage("hsv_img", hsv_img); cvNamedWindow("h_img", 0); //H通道 cvShowImage("h_img", h_img); cvNamedWindow("s_img", 0); //S通道 cvShowImage("s_img", s_img); cvNamedWindow("v_img", 0); //V通道 cvShowImage("v_img", v_img); cvWaitKey(0); cvReleaseImage(&hsv_img); cvReleaseImage(&h_img); cvReleaseImage(&s_img); cvReleaseImage(&v_img); cvDestroyWindow("hsv_img"); cvDestroyWindow("h_img"); cvDestroyWindow("s_img"); cvDestroyWindow("v_img"); return 0; }
最新发布
07-08

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值