边缘检测、Hough变换、轮廓提取、种子填充、轮廓跟踪

2023-10-28

转自:http://blog.sina.com.cn/s/blog_6c083cdd0100nm4s.html

 

7.1 边沿检测

我们给出一个模板 数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转) 和一幅图象 数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转) 。不难发现原图中左边暗,右边亮,中间存在着一条明显的边界。进行模板操作后的结果如下: 数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转)

可以看出,第3、4列比其他列的灰度值高很多,人眼观察时,就能发现一条很明显的亮边,其它区域都很暗,这样就起到了边沿检测的作用。

为什么会这样呢?仔细看看那个模板就明白了,它的意思是将右邻点的灰度值减左邻点的灰度值作为该点的灰度值。在灰度相近的区域内,这么做的结果使得该点的灰度值接近于0;而在边界附近,灰度值有明显的跳变,这么做的结果使得该点的灰度值很大,这样就出现了上面的结果。

这种模板就是一种边沿检测器,它在数学上的涵义是一种基于梯度的滤波器,又称边沿算子,你没有必要知道梯度的确切涵义,只要有这个概念就可以了。梯度是有方向的,和边沿的方向总是正交(垂直)的,例如,对于上面那幅图象的转置图象,边是水平方向的,我们可以用梯度是垂直方向的模板 数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转) 检测它的边沿。

例如,一个梯度为45度方向模板 数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转) ,可以检测出135度方向的边沿。

1.         Sobel算子

在边沿检测中,常用的一种模板是Sobel 算子。Sobel 算子有两个,一个是检测水平边沿的 数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转) ;另一个是检测垂直平边沿的 数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转) 。与 数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转)数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转) 相比,Sobel算子对于象素的位置的影响做了加权,因此效果更好。

Sobel算子另一种形式是各向同性Sobel(Isotropic Sobel)算子,也有两个,一个是检测水平边沿的 数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转) ,另一个是检测垂直平边沿的 数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转) 。各向同性Sobel算子和普通Sobel算子相比,它的位置加权系数更为准确,在检测不同方向的边沿时梯度的幅度一致。

下面的几幅图中,图7.1为原图;图7.2为普通Sobel算子处理后的结果图;图7.3为各向同性Sobel算子处理后的结果图。可以看出Sobel算子确实把图象中的边沿提取了出来。

数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转)

图7.1     原图

数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转)

图7.2     普通Sobel算子处理后的结果图

数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转)

图7.3     各向同性Sobel算子处理后的结果图

在程序中仍然要用到第3章介绍的通用3×3模板操作函数TemplateOperation,所做的操作只是增加几个常量标识及其对应的模板数组,这里就不再给出了。

2.         高斯拉普拉斯算子

由于噪声点(灰度与周围点相差很大的点)对边沿检测有一定的影响,所以效果更好的边沿检测器是高斯拉普拉斯(LOG)算子。它把我们在第3章中介绍的高斯平滑滤波器和拉普拉斯锐化滤波器结合了起来,先平滑掉噪声,再进行边沿检测,所以效果会更好。

常用的LOG算子是5×5的模板,如下所示 数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转) 。到中心点的距离与位置加权系数的关系用曲线表示为图7.4。是不是很象一顶墨西哥草帽?所以,LOG又叫墨西哥草帽滤波器。

数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转)

图7.4     LOG到中心点的距离与位置加权系数的关系曲线

图7.5为图7.1用LOG滤波器处理后的结果。

数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转)

图7.5     图7.1用LOG滤波器处理后的结果图

LOG的算法和普通模板操作的算法没什么不同,只不过把3×3改成了5×5,这里就不再给出了。读者可以参照第3章的源程序自己来完成。

7.2 Hough变换

Hough变化的原理是利用点和线的对偶性,将原始空间的给定的曲线通过曲线表达式变为参数空间的一个点。在原始图像坐标系下的一个点对应参数坐标系中的一条直线,同样参数坐标系的一条直线对应原始坐标中的一个点。原始坐标中呈现直线的所有点,它们的斜率和截距是相同的,所以他们在参数坐标下对应于同一个点。

首先,初始化一块缓冲徐,对应于参数平面,将所有的数据置0,对于图像的每一个前景点,求出参数平面对应的直线,把直线上所有的点都加1,最后找到参数平面最大的点的位置,这个位置就是原图像直线上的参数。

Hough变换用来在图象中查找直线。它的原理很简单:假设有一条与原点距离为s,方向角为θ的一条直线,如图7.6所示。

数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转)

图7.6    一条与原点距离为s,方向角为θ的一条直线

直线上的每一点都满足方程

数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转)

(7.1)

利用这个事实,我们可以找出某条直线来。下面将给出一段程序,用来找出图象中最长的直线(见图7.7)。找到直线的两个端点,在它们之间连一条红色的直线。为了看清效果,将结果描成粗线,如图7.8所示。

数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转)

图7.7 原图

数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转)

图7.8 Hough变换的结果

可以看出,找到的确实是最长的直线。方法是,开一个二维数组做为计数器,第一维是角度,第二维是距离。先计算可能出现的最大距离为 数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转) ,用来确定数组第二维的大小。对于每一个黑色点,角度的变化范围从00到1780(为了减少存储空间和计算时间,角度每次增加20而不是10),按方程(7.1)求出对应的距离s来,相应的数组元素[s][ 数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转) ]加1。同时开一个数组Line,计算每条直线的上下两个端点。所有的象素都算完后,找到数组元素中最大的,就是最长的那条直线。直线的端点可以在Line中找到。要注意的是,我们处理的虽然是二值图,但实际上是256级灰度图,不过只用到了0和255两种颜色。

BOOL Hough(HWND hWnd)

{

//定义一个自己的直线结构

    typedef struct{

                           int topx; //最高点的x坐标

                           int topy; //最高点的y坐标

                           int botx; //最低点的x坐标

                           int boty; //最低点的y坐标

                           }MYLINE;

       DWORD                             OffBits,BufSize;

     LPBITMAPINFOHEADER    lpImgData;

       LPSTR                     lpPtr;

       HDC                        hDc;

LONG                                  x,y;

       long                         i,maxd;

       int                           k;

       int                           Dist,Alpha;

HGLOBAL                  hDistAlpha,hMyLine;

     Int                                         *lpDistAlpha;

       MYLINE                             *lpMyLine,*TempLine,MaxdLine;

     static LOGPEN                rlp={PS_SOLID,1,1,RGB(255,0,0)};

     HPEN                              rhp;

//我们处理的实际上是256级灰度图,不过只用到了0和255两种颜色。

if( NumColors!=256){

       MessageBox(hWnd,"Must be a mono bitmap with grayscale palette!",

"Error Message",MB_OK|MB_ICONEXCLAMATION);

return FALSE;

}

//计算最大距离

       Dist=(int)(sqrt((double)bi.biWidth*bi.biWidth+

(double)bi.biHeight*bi.biHeight)+0.5);

       Alpha=180 /2 ;  //0 到 to 178 度,步长为2度

       //为距离角度数组分配内存

if((hDistAlpha=GlobalAlloc(GHND,(DWORD)Dist*Alpha*

sizeof(int)))==NULL){

MessageBox(hWnd,"Error alloc memory!","Error Message",

MB_OK|MB_ICONEXCLAMATION);

return FALSE;

    }

       //为记录直线端点的数组分配内存

      if((hMyLine=GlobalAlloc(GHND,(DWORD)Dist*Alpha*

sizeof(MYLINE)))==NULL){

            GlobalFree(hDistAlpha);

           return  FALSE;

       }

       OffBits=bf.bfOffBits-sizeof(BITMAPFILEHEADER);

//BufSize为缓冲区大小

       BufSize=OffBits+bi.biHeight*LineBytes;

     lpImgData=(LPBITMAPINFOHEADER)GlobalLock(hImgData);

     lpDistAlpha=(int *)GlobalLock(hDistAlpha);

       lpMyLine=(MYLINE *)GlobalLock(hMyLine);

for (i=0;i<(long)Dist*Alpha;i++){

              TempLine=(MYLINE*)(lpMyLine+i);

              (*TempLine).boty=32767; //初始化最低点的y坐标为一个很大的值

       }

       for (y=0;y<bi.biHeight;y++){

              //lpPtr指向位图数据

              lpPtr=(char *)lpImgData+(BufSize-LineBytes-y*LineBytes);

              for (x=0;x<bi.biWidth;x++)

                     if(*(lpPtr++)==0) //是个黑点

                            for (k=0;k<180;k+=2){

                                   //计算距离i

                              i=(long)fabs((x*cos(k*PI/180.0)+y*sin(k*PI/180.0)));

                                   //相应的数组元素加1

                              *(lpDistAlpha+i*Alpha+k/2)=*(lpDistAlpha+i*Alpha+k/2)+1;

                                   TempLine=(MYLINE*)(lpMyLine+i*Alpha+k/2);

                                   if(y> (*TempLine).topy){

                                          //记录该直线最高点的x,y坐标

                                          (*TempLine).topx=x;

                                          (*TempLine).topy=y;

                                   }

                                   if(y< (*TempLine).boty){

                                          //记录该直线最低点的x,y坐标

                                          (*TempLine).botx=x;

                                          (*TempLine).boty=y;

                                   }

                            }

}

       maxd=0;

       for (i=0;i<(long)Dist*Alpha;i++){

              TempLine=(MYLINE*)(lpMyLine+i);

              k=*(lpDistAlpha+i);

              if(k > maxd){

                     //找到数组元素中最大的,及相应的直线端点

                     maxd=k;

                     MaxdLine.topx=(*TempLine).topx;

                     MaxdLine.topy=(*TempLine).topy;

                     MaxdLine.botx=(*TempLine).botx;

                     MaxdLine.boty=(*TempLine).boty;

              }

       }

       hDc = GetDC(hWnd);

       rhp = CreatePenIndirect(&rlp);

       SelectObject(hDc,rhp);

       MoveToEx(hDc,MaxdLine.botx,MaxdLine.boty,NULL);

       //在两端点之间画一条红线用来标识

       LineTo(hDc,MaxdLine.topx,MaxdLine.topy);

      DeleteObject(rhp);                      

      ReleaseDC(hWnd,hDc);

       //释放内存及资源

       GlobalUnlock(hImgData);

GlobalUnlock(hDistAlpha);

       GlobalFree(hDistAlpha);

     GlobalUnlock(hMyLine);

       GlobalFree(hMyLine);

       return TRUE;

}

如果 数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转) 是给定的,用上述方法,我们可以找到该方向上最长的直线。

其实Hough变换能够查找任意的曲线,只要你给定它的方程。这里,我们就不详述了。

7.3 轮廓提取

轮廓提取的实例如图7.9、图7.10所示。

数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转)

图7.9     原图

数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转)

图7.10   轮廓提取

轮廓提取的算法非常简单,就是掏空内部点:如果原图中有一点为黑,且它的8个相邻点都是黑色时(此时该点是内部点),则将该点删除。要注意的是,我们处理的虽然是二值图,但实际上是256级灰度图,不过只用到了0和255两种颜色。源程序如下:

BOOL Outline(HWND hWnd)

{

       DWORD                             OffBits,BufSize;

     LPBITMAPINFOHEADER    lpImgData;

       LPSTR                   lpPtr;

HLOCAL                             hTempImgData;

       LPBITMAPINFOHEADER    lpTempImgData;

       LPSTR                   lpTempPtr;

       HDC                      hDc;

       HFILE                     hf;

       LONG                    x,y;

       int                                        num;

       int                               nw,n,ne,w,e,sw,s,se;

//我们处理的实际上是256级灰度图,不过只用到了0和255两种颜色。

if( NumColors!=256){

MessageBox(hWnd,"Must be a mono bitmap with grayscale palette!",

"Error Message",MB_OK|MB_ICONEXCLAMATION);

return FALSE;

}

OffBits=bf.bfOffBits-sizeof(BITMAPFILEHEADER);

//BufSize为缓冲区大小

       BufSize=OffBits+bi.biHeight*LineBytes;

       //为新图缓冲区分配内存

       if((hTempImgData=LocalAlloc(LHND,BufSize))==NULL)

   {

            MessageBox(hWnd,"Error alloc memory!","Error Message",MB_OK|

MB_ICONEXCLAMATION);

return FALSE;

    }

     lpImgData=(LPBITMAPINFOHEADER)GlobalLock(hImgData);   

       lpTempImgData=(LPBITMAPINFOHEADER)LocalLock(hTempImgData);

//拷贝头信息和位图数据

       memcpy(lpTempImgData,lpImgData,BufSize);

       for (y=1;y<bi.biHeight-1;y++){ //注意y的范围是从1到高度-2

              //lpPtr指向原图数据,lpTempPtr指向新图数据

              lpPtr=(char *)lpImgData+(BufSize-LineBytes-y*LineBytes);

              lpTempPtr=(char *)lpTempImgData+(BufSize-LineBytes-y*LineBytes);

              for (x=1;x<bi.biWidth-1;x++){

                     if(*(lpPtr+x)==0){ //是个黑点

                            //查找八个相邻点

                            nw=(unsigned char)*(lpPtr+x+LineBytes-1);

                            n=(unsigned char)*(lpPtr+x+LineBytes);

                            ne=(unsigned char)*(lpPtr+x+LineBytes+1);

                            w=(unsigned char)*(lpPtr+x-1);

                            e=(unsigned char)*(lpPtr+x+1);

                            sw=(unsigned char)*(lpPtr+x-LineBytes-1);

                            s=(unsigned char)*(lpPtr+x-LineBytes);

                            se=(unsigned char)*(lpPtr+x-LineBytes+1);

                            num=nw+n+ne+w+e+sw+s+se;

                            if(num==0) //说明都是黑点

                                   *(lpTempPtr+x)=(unsigned char)255; //删除该黑点

                     }

              }

       }

     if(hBitmap!=NULL)

           DeleteObject(hBitmap);

       hDc=GetDC(hWnd);    

       //创立一个新的位图

       hBitmap=CreateDIBitmap(hDc,(LPBITMAPINFOHEADER)lpTempImgData,

(LONG)CBM_INIT,

(LPSTR)lpTempImgData+

sizeof(BITMAPINFOHEADER)+

NumColors*sizeof(RGBQUAD),

(LPBITMAPINFO)lpTempImgData,

DIB_RGB_COLORS);

hf=_lcreat("c://outline.bmp",0);

       _lwrite(hf,(LPSTR)&bf,sizeof(BITMAPFILEHEADER));

       _lwrite(hf,(LPSTR)lpTempImgData,BufSize);

       _lclose(hf);

       //释放内存和资源

      ReleaseDC(hWnd,hDc);

       LocalUnlock(hTempImgData);

       LocalFree(hTempImgData);

       GlobalUnlock(hImgData);

       return TRUE;

}

7.4 种子填充

种子填充算法用来在封闭曲线形成的环中填充某中颜色,在这里我们只填充黑色。

种子填充其实上是图形学中的算法,其原理是:准备一个堆栈,先将要填充的点push进堆栈中;以后,每pop出一个点,将该点涂成黑色,然后按左上右下的顺序查看它的四个相邻点,若为白(表示还没有填充),则将该邻点push进栈。一直循环,直到堆栈为空。此时,区域内所有的点都被涂成了黑色。

这里,我们自己定义了一些堆栈的数据结构和操作,实现了堆栈的初始化、push、pop、判断是否为空、及析构。

//堆栈结构

typedef struct{

                HGLOBAL hMem; //堆栈全局内存句柄

               POINT *lpMyStack; //指向该句柄的指针

                LONG  ElementsNum; //堆栈的大小

                LONG  ptr; //指向栈顶的指针

                }MYSTACK;

//初始化堆栈的操作,第二个参数指定堆栈的大小

BOOL InitStack(HWND hWnd,LONG StackLen)

{

       SeedFillStack.ElementsNum=StackLen; //将堆栈的大小赋值

       if((SeedFillStack.hMem=GlobalAlloc(GHND,SeedFillStack.ElementsNum*

sizeof(POINT)))==NULL)

       {

              //内存分配错误,返回FALSE;

     MessageBox(hWnd,"Error alloc memory!","ErrorMessage",MB_OK|

MB_ICONEXCLAMATION);

              return FALSE;

       }

       SeedFillStack.lpMyStack=(POINT *)GlobalLock(SeedFillStack.hMem);

       //缓冲区全部清零

memset(SeedFillStack.lpMyStack,0,SeedFillStack.ElementsNum*

sizeof(POINT));

//堆顶指针为零

       SeedFillStack.ptr=0;

       //成功,返回TRUE

       return TRUE;

}

//析构函数

void DeInitStack()

{

       //释放内存,重置堆栈大小及栈顶指针。

       GlobalUnlock(SeedFillStack.hMem);

       GlobalFree(SeedFillStack.hMem);

       SeedFillStack.ElementsNum=0;

       SeedFillStack.ptr=0;

}

//push操作

BOOL MyPush(POINT p)

{

       POINT *TempPtr;

       if(SeedFillStack.ptr>=SeedFillStack.ElementsNum)

              return FALSE; //栈已满,返回FALSE

       //进栈,栈顶指针加1

       TempPtr=(POINT *)(SeedFillStack.lpMyStack+SeedFillStack.ptr++);

       (*TempPtr).x=p.x;

       (*TempPtr).y=p.y;

       return TRUE;

}

//pop操作

POINT MyPop()

{

       POINT InvalidP;

       InvalidP.x=-1;

       InvalidP.y=-1;

       if(SeedFillStack.ptr<=0)

              return InvalidP; //栈为空,返回无效点

       SeedFillStack.ptr--; //栈顶指针减1

       //返回栈顶点

       return *(SeedFillStack.lpMyStack+SeedFillStack.ptr);

}

//判断堆栈是否为空

BOOL IsStackEmpty()

{

       return (SeedFillStack.ptr==0)?TRUE:FALSE;

}

如果读者对堆栈的概念还不清楚,请参阅有关数据结构方面的书籍,这里就不详述了。

要注意的是:(1)要填充的区域是封闭的;(2)我们处理的虽然是二值图,但实际上是256级灰度图,不过只用到了0和255两种颜色;(3)在菜单中选择种子填充命令时,提示用户用鼠标点取一个要填充区域中的点,处理是在WM_LBUTTONDOWN中。

MYSTACK SeedFillStack;

BOOL SeedFill(HWND hWnd)

{

DWORD                   OffBits,BufSize;

     LPBITMAPINFOHEADER    lpImgData;

       HLOCAL                  hTempImgData;

       LPBITMAPINFOHEADER    lpTempImgData;

       LPSTR                   lpTempPtr,lpTempPtr1;

       HDC                                      hDc;

       HFILE                    hf;

       POINT                   CurP,NeighborP;

//我们处理的实际上是256级灰度图,不过只用到了0和255两种颜色。

       if( NumColors!=256){

MessageBox(hWnd,"Must be a mono bitmap with grayscale palette!",

"Error Message",MB_OK|MB_ICONEXCLAMATION);

return FALSE;

}

OffBits=bf.bfOffBits-sizeof(BITMAPFILEHEADER);

//BufSize为缓冲区大小

       BufSize=OffBits+bi.biHeight*LineBytes;

//为新图缓冲区分配内存

       if((hTempImgData=LocalAlloc(LHND,BufSize))==NULL)

   {

            MessageBox(hWnd,"Error alloc memory!","Error Message",MB_OK|

MB_ICONEXCLAMATION);

return FALSE;

    }

     lpImgData=(LPBITMAPINFOHEADER)GlobalLock(hImgData);   

       lpTempImgData=(LPBITMAPINFOHEADER)LocalLock(hTempImgData);

//拷贝头信息和位图数据

       memcpy(lpTempImgData,lpImgData,BufSize);

       if(!InitStack(hWnd,(LONG)bi.biHeight*bi.biWidth)){  //初始化堆栈

              //若失败,释放内存,返回

              LocalUnlock(hTempImgData);

              LocalFree(hTempImgData);

              GlobalUnlock(hImgData);

              return FALSE;

       }

       lpTempPtr=(char*)lpTempImgData+

(BufSize-LineBytes-SeedPoint.y*LineBytes)+SeedPoint.x;

       if(*lpTempPtr==0){

              //鼠标点到了黑点上,提示用户不能选择边界上的点,返回FALSE

MessageBox(hWnd,"The point you select is a contour point!",

"Error Message",MB_OK|MB_ICONEXCLAMATION);

              LocalUnlock(hTempImgData);

              LocalFree(hTempImgData);

              GlobalUnlock(hImgData);

              DeInitStack();

return FALSE;

       }

       //push该点(用户用鼠标选择的,处理是在WM_LBUTTONDOWN中

       MyPush(SeedPoint);

       while(!IsStackEmpty()) //堆栈不空则一直处理

       {

              CurP=MyPop(); //pop栈顶的点

              lpTempPtr=(char*)lpTempImgData+

(BufSize-LineBytes-CurP.y*LineBytes)+CurP.x;

              //将该点涂黑

              *lpTempPtr=(unsigned char)0;

              //左邻点

              if(CurP.x>0) //注意判断边界

              {

                     NeighborP.x=CurP.x-1;

                     NeighborP.y=CurP.y;

                     lpTempPtr1=lpTempPtr-1;

                     if(*lpTempPtr1!=0) //如果为白,表示还没有填,进栈

                            MyPush(NeighborP);

              }

//上邻点

              if(CurP.y>0) //注意判断边界

              {

                     NeighborP.x=CurP.x;

                     NeighborP.y=CurP.y-1;

                     lpTempPtr1=lpTempPtr+LineBytes;

                     if(*lpTempPtr1!=0) //如果为白,表示还没有填,进栈

                            MyPush(NeighborP);

              }

//右邻点

              if(CurP.x<bi.biWidth-1) //注意判断边界

              {

                     NeighborP.x=CurP.x+1;

                     NeighborP.y=CurP.y;

                     lpTempPtr1=lpTempPtr+1;

                     if(*lpTempPtr1!=0) //如果为白,表示还没有填,进栈

                            MyPush(NeighborP);

              }

              //下邻点

              if(CurP.y<bi.biHeight-1) //注意判断边界

              {

                     NeighborP.x=CurP.x;

                     NeighborP.y=CurP.y+1;

                     lpTempPtr1=lpTempPtr-LineBytes;

                     if(*lpTempPtr1!=0) //如果为白,表示还没有填,进栈

                            MyPush(NeighborP);

              }

       }

       //析构堆栈,释放内存

       DeInitStack();

if(hBitmap!=NULL)

           DeleteObject(hBitmap);

       hDc=GetDC(hWnd);    

       //创建新的位图

       hBitmap=CreateDIBitmap(hDc,(LPBITMAPINFOHEADER)lpTempImgData,

(LONG)CBM_INIT,

(LPSTR)lpTempImgData+

sizeof(BITMAPINFOHEADER)+

NumColors*sizeof(RGBQUAD),

(LPBITMAPINFO)lpTempImgData,

DIB_RGB_COLORS);

       hf=_lcreat("c://seed.bmp",0);

       _lwrite(hf,(LPSTR)&bf,sizeof(BITMAPFILEHEADER));

       _lwrite(hf,(LPSTR)lpTempImgData,BufSize);

       _lclose(hf);

       //释放内存和资源

      ReleaseDC(hWnd,hDc);

       LocalUnlock(hTempImgData);

       LocalFree(hTempImgData);

       GlobalUnlock(hImgData);

       return TRUE;

}

7.5 轮廓跟踪

轮廓跟踪,顾名思义就是通过顺序找出边缘点来跟踪出边界。图7.9经轮廓跟踪后得到的结果如图7.11所示。

数字图像处理笔记2- <wbr>边沿检测与提取,轮廓跟踪(转)

图7.11    图7.9轮廓跟踪后的结果

一个简单二值图象闭合边界的轮廓跟踪算法很简单:首先按从上到下,从左到右的顺序搜索,找到的第一个黑点一定是最左上方的边界点,记为A。它的右,右下,下,左下四个邻点中至少有一个是边界点,记为B。从开始B找起,按右,右下,下,左下,左,左上,上,右上的顺序找相邻点中的边界点C。如果C就是A点,则表明已经转了一圈,程序结束;否则从C点继续找,直到找到A为止。判断是不是边界点很容易:如果它的上下左右四个邻居都是黑点则不是边界点,否则是边界点。源程序如下,其中函数IsContourP用来判断某点是不是边界点。

BOOL Contour(HWND hWnd)

{

       DWORD                             OffBits,BufSize;

LPBITMAPINFOHEADER    lpImgData;

       LPSTR                   lpPtr;

       HLOCAL                  hTempImgData;

       LPBITMAPINFOHEADER    lpTempImgData;

       LPSTR                   lpTempPtr;

       HDC                      hDc;

       HFILE                    hf;

       LONG                    x,y;

       POINT                   StartP,CurP;

       BOOL                     found;

       int                        i;

int       direct[8][2]={{1,0},{1,1},{0,1},{-1,1},{-1,0},{-1,-1},{0,-1},{1,-1}};

//我们处理的实际上是256级灰度图,不过只用到了0和255两种颜色。

       if( NumColors!=256){

MessageBox(hWnd,"Must be a mono bitmap with grayscale palette!",

"Error Message",MB_OK|MB_ICONEXCLAMATION);

return FALSE;

}

//到位图数据的偏移值

       OffBits=bf.bfOffBits-sizeof(BITMAPFILEHEADER);

       //缓冲区大小

BufSize=OffBits+bi.biHeight*LineBytes;

//为新图缓冲区分配内存

       if((hTempImgData=LocalAlloc(LHND,BufSize))==NULL)

     {

       MessageBox(hWnd,"Error alloc memory!","Error Message",

MB_OK|MB_ICONEXCLAMATION);

return FALSE;

}

      lpImgData=(LPBITMAPINFOHEADER)GlobalLock(hImgData);

       lpTempImgData=(LPBITMAPINFOHEADER)LocalLock(hTempImgData);

       //新图缓冲区初始化为255

       memset(lpTempImgData,(BYTE)255,BufSize);

       //拷贝头信息

       memcpy(lpTempImgData,lpImgData,OffBits);

       //找到标志置为假

       found=FALSE;

       for (y=0;y<bi.biHeight && !found; y++){

              lpPtr=(char *)lpImgData+(BufSize-LineBytes-y*LineBytes);

              for (x=0;x<bi.biWidth && !found; x++)

                     if (*(lpPtr++) ==0) found=TRUE;

//找到了最左上的黑点,一定是个边界点

       }

       if(found){ //如果找到了,才做处理

//从循环退出时,x,y坐标都做了加1的操作。在这里把它们减1,得到

//起始点坐标StartP

              StartP.x=x-1;

              StartP.y=y-1;

              lpTempPtr=(char*)lpTempImgData+

(BufSize-LineBytes-StartP.y*LineBytes)+StartP.x;

              *lpTempPtr=(unsigned char)0; //起始点涂黑

              //右邻点

            CurP.x=StartP.x+1;

              CurP.y=StartP.y;

              lpPtr=(char *)lpImgData+(BufSize-LineBytes-CurP.y*LineBytes)+CurP.x;

              if(*lpPtr!=0){ //若右邻点为白,则找右下邻点

                   CurP.x=StartP.x+1;

                     CurP.y=StartP.y+1;

                     lpPtr=(char*)lpImgData+

(BufSize-LineBytes-CurP.y*LineBytes)+CurP.x;

                     if(*lpPtr!=0){ //若仍为白,则找下邻点

                          CurP.x=StartP.x;

                            CurP.y=StartP.y+1;

                     }

                     else{ //若仍为白,则找左下邻点

                          CurP.x=StartP.x-1;

                            CurP.y=StartP.y+1;

                     }

              }

              while (! ( (CurP.x==StartP.x) &&(CurP.y==StartP.y))){ //知道找到起始点,

//循环才结束

                     lpTempPtr=(char*)lpTempImgData+

(BufSize-LineBytes-CurP.y*LineBytes)+CurP.x;

                     *lpTempPtr=(unsigned char)0;

                     for(i=0;i<8;i++){

//按右,右上,上,左上,左,左下,下,右下的顺序找相邻点

//direct[i]中存放的是该方向x,y的偏移值

                            x=CurP.x+direct[i][0];

                            y=CurP.y+direct[i][1];

              //lpPtr指向原图数据,lpTempPtr指向新图数据

                            lpTempPtr=(char*)lpTempImgData+

(BufSize-LineBytes-y*LineBytes)+x;

                            lpPtr=(char *)lpImgData+(BufSize-LineBytes-y*LineBytes)+x;

                            if(((*lpPtr==0)&&(*lpTempPtr!=0))||

((x==StartP.x)&&(y==StartP.y)))

                            //原图中为黑点,且新图中为白点(表示还没搜索过)时才处理

                            //另一种可能是找到了起始点

                                   if(IsContourP(x,y,lpPtr)){ //若是个边界点

                                          //记住当前点的位置

                            CurP.x=x;

                                          CurP.y=y;

                                          break;

                                   }

                     }

              }

       }

    if(hBitmap!=NULL)

           DeleteObject(hBitmap);

       hDc=GetDC(hWnd);

       //创立一个新的位图

       hBitmap=CreateDIBitmap(hDc,(LPBITMAPINFOHEADER)lpTempImgData,

(LONG)CBM_INIT,

(LPSTR)lpTempImgData+

sizeof(BITMAPINFOHEADER)+

NumColors*sizeof(RGBQUAD),

                                  (LPBITMAPINFO)lpTempImgData,

DIB_RGB_COLORS);

       hf=_lcreat("c://contour.bmp",0);

       _lwrite(hf,(LPSTR)&bf,sizeof(BITMAPFILEHEADER));

       _lwrite(hf,(LPSTR)lpTempImgData,BufSize);

       _lclose(hf);

       //释放内存和资源

      ReleaseDC(hWnd,hDc);

       LocalUnlock(hTempImgData);

       LocalFree(hTempImgData);

       GlobalUnlock(hImgData);

       return TRUE;

}

//判断某点是不是边界点,参数x,y 为该点坐标,lpPtr为指向原数据的指针

BOOL IsContourP(LONG x,LONG y, char *lpPtr)

{

       int    num,n,w,e,s;

       n=(unsigned char)*(lpPtr+LineBytes); //上邻点

       w=(unsigned char)*(lpPtr-1); //左邻点

       e=(unsigned char)*(lpPtr+1); //右邻点

       s=(unsigned char)*(lpPtr-LineBytes); //下邻点

       num=n+w+e+s;

       if(num==0) //全是黑点,说明是个内部点而不是边界点

              return FALSE;

       return TRUE;

 

}

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

边缘检测、Hough变换、轮廓提取、种子填充、轮廓跟踪 的相关文章

  • 禁用按钮上的实际文本颜色

    VB2012 我正在创建一个按钮控件并继承自 NET 按钮 从这里夺取基地https blogs msdn microsoft com jfoscoding 2005 11 10 building a splitbutton https b
  • .NET - 将颜色名称字符串转换为 System.Drawing.Color

    将 red green yellow aliceblue 等字符串转换为实际的 System Drawing Color 值的最佳方法是什么 我正在查看反思 发现有些事情似乎不对劲 您可以使用 Color FromName
  • 将 RGB 转换为灰度/强度

    当从 RGB 转换为灰度时 据说应该对 R G 和 B 通道应用特定的权重 这些权重是 0 2989 0 5870 0 1140 据说这是因为人类对这三种颜色的感知 感受不同 有时也有人说这些是用于计算 NTSC 信号的值 但是 我在网上没
  • 无法索引空数组

    我正在使用一个模板 该模板根据服务器备份是否成功的条件设置单元格颜色 我有下面的代码 它不断抛出错误 无法索引到空数组 Cannot index into a null array At C Users admin Desktop new
  • 如何在 CSS 中使用 3 位颜色代码而不是 6 位颜色代码?

    我最近检查了我的 CSS 文件 并将所有六位十六进制代码转换为简单的三位数代码 例如 我的 FDFEFF被缩短为 FFF 它呈现的颜色与以前几乎完全相同 在我看来 中间部分相当无用 删除它们在我的 CSS 文件中节省了整整 300 个字节
  • 空合并运算符分配给 self

    我目前在 Unity 中设置了两个脚本来处理一些 UI 音频 一个是管理器 另一个是为特定 UI 元素播放声音 我所拥有的简化版本是这样的 public class AudioUIManager MonoBehaviour Only one
  • 如何使用PIL将灰度图转为伪彩色?

    我似乎不知道如何获取我的灰度函数并将其更改为给我假颜色 我知道我需要将每种颜色 R G B 分成范围 然后根据每种颜色的范围分配颜色 有谁知道这是如何运作的 def grayscale pic width height pic size f
  • Mercurial 颜色扩展的有效颜色是什么?

    水星的color 扩大 http mercurial selenic com wiki ColorExtension很棒 但是该页面上没有有效颜色的列表 我寻找了一个名为 color 的 python 文件 希望能够阅读源代码并在那里看到它
  • 如何在 Go 中表示可选字符串?

    我希望建模一个可以有两种可能形式的值 不存在或字符串 执行此操作的自然方法是Maybe String or Optional
  • 查询外键列可以为NULL的地方

    我想获取数据 如果orgid 2或者如果根本没有行uid orgid is an integer 我能想到的最接近的事情就是做IS NULL但我没有得到数据uid没有一个orgid排 任何想法 select u uid u fname u
  • 需要帮助从数组中为国家/地区着色,保留其余默认颜色

    我需要一些帮助从我创建的数组中获取数据 然后仅对数组中存在的国家 地区进行着色 而不在数组中的其余国家 地区我希望保留为默认颜色 我正在使用 D3 来完成所有这些工作 并且我非常确定我可以通过 D3 实现我需要的目标 但不确定如何实现 我想
  • Material UI - 更改主题中的按钮文本颜色

    我在直接在 Material UI 主题中更改按钮文本颜色时遇到问题 更改主色 按钮字体大小效果很好 因此问题不在于传递主题 这是我的代码 import React from react import MuiThemeProvider cr
  • 如何使用 Robot Framework 在控制台输出中添加颜色

    我想在 RobotFramework 的控制台输出中添加一些颜色 我尝试使用控制台颜色代码 例如 message Set Variable hello world Log To Console e 0 36 49m message e 0
  • Java JScrollBar设计

    我想自定义 JScrollBar 设计 我使用 Mac 使用 eclipse 开发应用程序 我已经尝试过scrollPane getVerticalScrollBar setBackground Color BLACK 但什么也没发生 我的
  • SQL中的NULL和编程语言中的NULL之间的区别

    我刚刚遇到一个关于如何在 T SQL 可能还有其他形式的 SQL 中处理 NULL 的有趣场景 这个问题得到了很好的描述和回答这个问题 https stackoverflow com questions 2866714 how does a
  • 默认情况下,Spark sql 模式中的可为空性是建议性的。严格执行的最佳方法是什么?

    我正在开发一个简单的 ETL 项目 它读取 CSV 文件 执行 对每列进行一些修改 然后将结果以 JSON 格式写出 我想要读取我的结果的下游进程 确信我的输出符合 一个商定的模式 但我的问题是 即使我定义 我的输入模式的所有字段都为 nu
  • 使用滤镜将css3灰色图像转为蓝色?

    我正在尝试将灰色图像变为更蓝色的色调 真的不知道如何为此设置滤镜或是否可能 该图像只有一种颜色 cacaca 其余部分透明 我正在尝试使用相同的图像进行一些叠加 以便它仅突出显示那些彩色部分而不是透明区域 一直在尝试其中的一些 但没有取得多
  • 将 NULL 变量插入数据库

    我将变量设置为 NULL 我试图将其插入数据库 但由于某种原因 它们一直以 0 的形式提交 我确信我试图插入的列允许 NULL 并且默认值设置为 NULL 这是我的代码 insert NULL query mysql query INSER
  • 如何给 Git 控制台着色?

    我最近看到gitWindows 中的控制台是彩色的 例如绿色表示添加 红色表示删除等 我如何为我的颜色上色git这样的控制台 为了安装它 我使用了以下命令 sudo apt get install git core As noted htt
  • R:如何改变格子(levelplot)颜色主题?

    我安装的默认主题是将值映射为粉色和青色的主题 例如如何将其更改为灰度主题 您可以使用 library lattice lattice options default theme standard theme color FALSE 它会打开

随机推荐

  • InitializingBean讲解

    InitializingBean讲解 Spring中有两种类型的Bean 一种是普通Bean 另一种是工厂Bean 即FactoryBean 工厂Bean跟普通Bean不同 其返回的对象不是指定类的一个实例 其返回的是该工厂Bean的get
  • cx_Oracle使用方法

    正确安装好cx oracle之后 要使用它来连接到oracle数据库进行操作 具体应该分3步走 第一步 导入cx Oracle 建立连接 gt gt gt import cx Oracle 导入模块 gt gt gt db cx Oracl
  • SpringBoot入门 快速创建并部署web后端

    这两天学习了springboot些框架开发 发现用它开发真的是简单便捷 就像是它的设计初衷所描述的那样 它虽然没有提出任何新的技术 但却将之前的spring技术集成了 他让spring变得更加好用 于是将最近的学习内容总结一下 分享给大家
  • 基于MATLAB的线激光三维彩色扫描仪

    暑期做的一个项目 开始并不是很熟悉 在网上查找的资料也不是很具体 但是自习学习了理论知识之后还是比较容易的做出来这个项目 现在开源整个项目 由于篇幅有限 本文适合稍微有点点基础的朋友 源码见底部 先显示下最后结果 一 硬件设计 主要有步进电
  • struts2与hibernate整合遇到的问题

    1 问题描述 今天练习struts2和hibernate整合 结果各层都写好了浏览器还出现了service实例空指针的错误 控制台只是提示没有找到什么值栈啊什么的 解决过程 查看各层代码 配置文件是否写的有问题 都没问题 后来就找是不是包的
  • 如何登录GItHub

    1 找到hosts 2 用管理员权限运行 3 打开终端 4 执行cmd 5 再执行notepad hosts 6 会自动弹出hosts 7 查找3个需要添加在hosts最后的内容 1 github com IP地址查询 https gith
  • Pandas常用函数操作示例

    一 概述 本文主要记录一些常用的pandas 操作示例 可收藏用作日常编码中的速查手册 用到的示例可以在下面的索引分类中找到 二 目录 文章目录 一 概述 二 目录 三 示例 1 pandas 创建 Series 通过 list 创建 se
  • CentOS7 学习 10 常用指令 7 压缩、解压缩

    索引 gzip 压缩 gunzip 解压 zip 压缩 unzip 解压 在项目打包发布中常用 这个和windows一样 tar zcvf xx tar gz 要压缩的文件file tar zxvf xx tar gz C 解压到的路径di
  • ELK日志收集系统

    Kibana 数据分析工具 提供数据的聚合 分析功能 数据的可视化 用图表展现数据 分析图标可进行分享 或在web应用中引用图表 Kibana汉化 docker cp kibana usr share kibana config kiban
  • 一篇“从入门到上手”的PCB设计教程

    一篇 从入门到上手 的PCB设计教程 这是一篇面向神马都不懂的小白玩家的PCB设计教程 希望能帮助大家快速上手PCB的设计 1 预备知识 1 1 常用工具 1 做图工具 Altium Designer 2 PCB板加工 嘉立创 3 元件封装
  • react的onClick自动触发等相关问题

    react分页组件遇到的问题 private getFirst const pageNo this state if pageNo gt 3 return span 首页 span else return private changePag
  • 基于openlayers的最短路径规划

    之前的文章讲到了如何构建空间数据库 矢量数据如何入库 如何构建拓扑网络 如何自定义查询函数 如何构建wms服务 本文讲解如何基于openlayers晚上最短路径规划功能 一 基于openlayers3 1 构建网页 这里只是一个简单的网页
  • idea 创建mybatis xml文件时找不到

    1 File gt Settings 如图 2 添加模板 如下图 3 添加xml模板 如下图 模板内容
  • 编译openwrt全过程(超详细)

    本教程的编译环境 win7 专业版 VMwareWorkstation6 5虚拟机 Ylmf OS 3 0 编译的过程中要保持电脑联网 搭建编译环境 应用程序 附件 终端 sudo apt get update 更新 安装编译需要的组件 s
  • 前端最新一面

    vue2 vue3 v model的区别 instance of原理 事件循环机制打印顺序 打印结果 1 3
  • 手机上编写Java程序的软件

    对于程序员来说 编写代码几乎都是在电脑上 但有时候在一些特殊情况下 没有电脑 或者不方便带电脑 这时就想 要是能在手机上写代码该多好啊 以前我也折腾过 找过许多软件 但感觉不如我意 但我并没有放弃 在浏览YouTube的时候 偶然发现了一款
  • 低功耗技术(三)UPF的使用

    UPF是一个统一的 被广泛应用的低功耗实现标准 它用一些标准的语言描述用户的低功耗设计意图 一 UPF所需要的特殊单元库 1 Level Shifter和Isolation Cell 对于多电压设计 需要用Level shifter来实现不
  • SpringBoot整合JWT和MD5实现单点登录

    1 知识点 1 1 JWT JSON Web Token JSON Web令牌 是一个开放标准 rfc7519 它定义了一种紧凑的 自包含的方式 用于在各方之间以JSON对象安全地传输信息 此信息可以验证和信任 因为它是数字签名的 jwt可
  • Android开发实践:Java层与Jni层的数组传递

    http www linuxidc com Linux 2014 03 97561 htm Android开发中 经常会在Java代码与Jni层之间传递数组 byte 一个典型的应用是Java层把需要发送给客户端的数据流传递到Jni层 由J
  • 边缘检测、Hough变换、轮廓提取、种子填充、轮廓跟踪

    转自 http blog sina com cn s blog 6c083cdd0100nm4s html 7 1 边沿检测 我们给出一个模板 和一幅图象 不难发现原图中左边暗 右边亮 中间存在着一条明显的边界 进行模板操作后的结果如下 可