opencv实现连通域

分享一下我老师大神的人工智能教程!零基础,通俗易懂!http://blog.csdn.net/jiangjunshow

也欢迎大家转载本篇文章。分享知识,造福人民,实现我们中华民族伟大复兴!

               

      在本文中使用图像连通域统计使用opencv中的cvFloodFill方法,但是在cvFloodFill方法中CvConnectedComp参数无法返回具体点坐标位置信息,找了些资料、给CvSeq分配空间但是还是contour内容没有值,估计是OPENCV2.00版本不支持。如果想获取点坐标信息有两种方法:一是通过源代码改写cvFloodFill函数来返回点坐标信息;二是cvFloodFill连通之后,根据返回区域统计多少个点,并统计之后对此点做标记,依次反复下一个连通区域。
   以下是方法二的代码实现:

//连通区域结构
typedef struct MyConnectedComp
{
 double area;
 int value;
 RECT rect;
 list<POINT> listAllPoint;
}MyConnectedComp
/*
*功能说明:统计图像的连通区域
*参数说明:src,表示原始图象;rectSrc,表示图像区域;listData,表示连通域结构链表;nUnionColor,表示连通颜色(0,表示黑点;1,表示白点);nUnionMode,表示连通类型(0,表示4连通;1,表示8连通)
*返回值:int类型。-1,表示参数错误;0,表示非二值/灰度图像
*/
int ImageUnionByFloodFill(CxImage *src,RECT rectSrc,list<MyConnectedComp> &listData,int nUnionColor,int nUnionMode)
{
 int nRet = 1;
 //参数检测
 if(src==NULL  || !(nUnionColor==0||nUnionColor==1) || !(nUnionMode==1||nUnionMode==2) || (rectSrc.left==0 && rectSrc.right==0 || rectSrc.top==0 && rectSrc.bottom==0 || rectSrc.left>rectSrc.right || rectSrc.top>rectSrc.bottom))
 {
  nRet = -1;
  return nRet;
 }
 if(src->GetBpp()>8)
 {
  nRet = 0;
  return  nRet;
 }
 listData.clear();
 int nColor = 255;
 if(nUnionColor==0)
 {
  nUnionColor = 255;
 }
 if(nUnionColor==1)
 {
  nUnionColor = 0;
 }
 int nFlag = 4;
 if(nUnionMode==1)
 {
  nFlag = 4;
 }
 if(nUnionMode==2)
 {
  nFlag = 8;
 }
 int iBackColor = GetBlackColor(*src);
 long i = 0,j = 0;
 long m = 0,n = 0,newN = 0,newM = 0;
 long nWidth = 0,nHeight = 0;

 IplImage *pcvImageTmp = NULL;
 if(!Cximage2IplImage(src,&pcvImageTmp))
 {
  return 0;
 }
 //如果是灰度图像,做二值化处理
 if(src->GetBpp()!=1)
 {
  cvThreshold(pcvImageTmp,pcvImageTmp,128,255,CV_THRESH_BINARY);
 }
 CvSize size = cvGetSize(pcvImageTmp);
 CvConnectedComp dccTmp;

CvConnectedComp comp;

 CvPoint seed;
 for(j=rectSrc.top;j<rectSrc.bottom;j++)
 {
  for(i=rectSrc.left;i<rectSrc.right;i++)
  {
   if(i<0 || i>=size.width || j<0 || j>=size.height)
   {
    continue;
   }
   uchar *p = (uchar *)&(pcvImageTmp->imageData+j*pcvImageTmp->widthStep)[i]; 
   if(*p==nUnionColor || *p==100 || *p==101)
   {
    continue;
   }
   //处理为黑点,做连通
   seed = cvPoint(i,j);
   cvFloodFill(pcvImageTmp, seed,cvScalarAll(100), cvScalarAll(0),cvScalarAll(0), &comp, nFlag, 0 );  
   dccTmp.area = comp.area;
   dccTmp.rect.left = comp.rect.x;
   dccTmp.rect.top = comp.rect.y;
   dccTmp.rect.right = comp.rect.x + comp.rect.width;
   dccTmp.rect.bottom = comp.rect.y + comp.rect.height;
   dccTmp.value = int(comp.value.val[1]);
   dccTmp.listAllPoint.clear();
   POINT ptInsert;
   for(n=0;n<=comp.rect.height;n++)
   {
    for(m=0;m<comp.rect.width;m++)
    {
     newN = n + comp.rect.y;
     newM = m + comp.rect.x;
     if(newN>=size.height || newM>=size.width)
     {
      continue;
     }
     uchar *q = (uchar *)&(pcvImageTmp->imageData+newN*pcvImageTmp->widthStep)[newM]; 
     if(*q==100)
     {
      ptInsert.x = newM;
      ptInsert.y = newN;
      dccTmp.listAllPoint.push_back(ptInsert);
      *q = 101;//改变标记,说明已被连通
     }
    }
   }
   listData.push_back(dccTmp);
  }
 }
 cvReleaseImage(&pcvImageTmp);
 return nRet;
}

注:CxImage是一款优秀的图像操作类库。它可以快捷地存取、显示、转换各种图像。

下载地址:http://www.codeproject.com/KB/graphics/cximage.aspx

           

给我老师的人工智能教程打call!http://blog.csdn.net/jiangjunshow
这里写图片描述
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值