Canny边缘检测原理及C#程序实现

http://blog.csdn.net/yjz_uestc/article/details/6664937

Canny边缘检测是被公认的检测效果最好的边缘检测方法,是由John F. Canny于1986年提出,算法目标是找出一个最优的边缘检测的方法,所谓最优即:1.好的检测:算法能够尽可能的标识出图像的边缘;2.好的定位:标识出的边缘要尽可能的与实际边缘相接近;3.最小响应:图像中的边缘只能标识一次,并且不能把噪声标识成边缘。同时我们也要满足3个准则:信噪比准则、定位精度准则、单边缘响应准则。

Canny边缘检测算法可分为4步:

高斯滤波器平滑、计算梯度、非极大值抑制、双阈值边缘检测和边缘连接。

(经典不会随着时间褪色,算法也是一样)

下面将逐步讲解并给出程序:

第一步:高斯平滑

为什么要对图像(灰度图像)进行高斯平滑预处理呢?高斯滤波器对去除服从正态分布的的噪声很有效,我做过实验,随着高斯模板的增大,被识别的边缘会逐渐减少,所以通过选着适合大小的高斯模板平滑,可以比较有效的去除一些伪边缘点。

第二步:计算梯度

首先,由一阶导数算子(一般用sobel模板)计算灰度图像每个像素点在水平和竖直方向上的导数Gx、Gy,得出梯度向量(Gx,Gy),计算梯度的值G和方向theta:

G=sqrt(Gx*Gx+Gy*Gy)  theta=arctan(Gy/Gx)

然后,将每个像素点的梯度的值和方向分别放入两个数组中,程序如下:

[csharp] view plaincopy

  1. <span style="font-size:16px;">byte[] orients = new byte[width * height];// 梯度方向数组
  2. float[,] gradients = new float[width, height];// 梯度值数组
  3. double gx, gy;
  4. for (int i = 1; i < (height - 1);i++ )
  5. {
  6. for (int j = 1; j < (width - 1); j++)
  7. {
  8. //求水平和竖直导数
  9. gx = bufdata[(i - 1) * width + j] + bufdata[(i + 1) * width + j] - bufdata[(i -1) * width + j - 1] - bufdata[(i + 1) * width + j - 1]+ 2*(bufdata[i * width + j + 1] - bufdata[i * width + j - 1]);
  10. gy = bufdata[(i - 1) * width + j - 1] + bufdata[(i + 1) * width + j + 1] - bufdata[(i + 1) * width + j - 1] - bufdata[(i + 1) * width + j + 1]+ 2*(bufdata[(i - 1) * width + j] - bufdata[(i + 1) * width + j - 1]);
  11. gradients[j, i] = (float)Math.Sqrt(gx * gx + gy * gy);
  12. if (gx == 0)
  13. {
  14. orientation = (gy == 0) ? 0 : 90;
  15. }
  16. else
  17. {
  18. double div = (double)gy / gx;
  19. if (div < 0)
  20. {
  21. orientation = 180 - Math.Atan(-div) * toAngle;
  22. }
  23. else
  24. {
  25. orientation = Math.Atan(div) * toAngle;
  26. }
  27. //只保留成4个方向
  28. if (orientation < 22.5)
  29. orientation = 0;
  30. else if (orientation < 67.5)
  31. orientation = 45;
  32. else if (orientation < 112.5)
  33. orientation = 90;
  34. else if (orientation < 157.5)
  35. orientation = 135;
  36. else orientation = 0;
  37. }
  38. orients[i*width+j] = (byte)orientation;
  39. }
  40. } </span>

第三步:非极大值抑制

如果直接把梯度作为边缘的话,将得到一个粗边缘的图像,这不满足上面提到的准则,我们希望得到定位准确的单像素的边缘,所以将每个像素点的梯度与其梯度方向上的相邻像素比较,如果不是极大值,将其置0,否则置为某一不大于255的数,程序如下:

[csharp] view plaincopy

  1. <span style="font-size:16px;"> float leftPixel = 0, rightPixel = 0;
  2. for (int y = 1; y <height-1; y++)
  3. {
  4. for (int x = 1; x < width-1; x++)
  5. {
  6. //获得相邻两像素梯度值
  7. switch (orients[y * width + x])
  8. {
  9. case 0:
  10. leftPixel = gradients[x - 1, y];
  11. rightPixel = gradients[x + 1, y];
  12. break;
  13. case 45:
  14. leftPixel = gradients[x - 1, y + 1];
  15. rightPixel = gradients[x + 1, y - 1];
  16. break;
  17. case 90:
  18. leftPixel = gradients[x, y + 1];
  19. rightPixel = gradients[x, y - 1];
  20. break;
  21. case 135:
  22. leftPixel = gradients[x + 1, y + 1];
  23. rightPixel = gradients[x - 1, y - 1];
  24. break;
  25. }
  26. if ((gradients[x, y] < leftPixel) || (gradients[x, y] < rightPixel))
  27. {
  28. dis[y * disdata.Stride + x] = 0;
  29. }
  30. else
  31. {
  32. dis[y * disdata.Stride + x] = (byte)(gradients[x, y] /maxGradient* 255);//maxGradient是最大梯度
  33. }
  34. }
  35. }   </span>

第四步:双阈值边缘检测

由上一步得到的边缘还有很多伪边缘,我们通过设置高低双阈值的方法去除它们,具体思想是:梯度值大于高阈值的像素点认为其一定是边缘,置为255,梯度值小于低阈值的像素点认为其一定不是边缘置为0,介于两阈值之间的点像素点为待定边缘。然后,考察这些待定边缘点,如果其像素点周围8邻域的梯度值都小于高阈值,认为其不是边缘点,置为0;至于,如何设定双阈值大小,我们可以根据实际情况设定,如设成100和20,也可以根据图像梯度值的统计信息设定,一般小阈值是大阈值的0.4倍即可。程序如下:

[csharp] view plaincopy

  1. <span style="font-size:16px;">fmean = fmean / maxGradient * 255;//某统计信息
  2. highThreshold = (byte)(fmean);//高阈值
  3. lowThreshold = (byte)(0.4 * highThreshold); //低阈值
  4. for (int y = 0; y < height; y++)
  5. {
  6. for (int x = 0; x < width; x++)
  7. {
  8. if (dis[y * disdata.Stride + x] < highThreshold)
  9. {
  10. if (dis[y * disdata.Stride + x] < lowThreshold)
  11. {
  12. dis[y * disdata.Stride + x] = 0;
  13. }
  14. else
  15. {
  16. if ((dis[y * disdata.Stride + x - 1] < highThreshold) &&
  17. (dis[y * disdata.Stride + x + 1] < highThreshold) &&
  18. (dis[(y - 1) * disdata.Stride + x - 1] < highThreshold) &&
  19. (dis[(y - 1) * disdata.Stride + x] < highThreshold) &&
  20. (dis[(y - 1) * disdata.Stride + x + 1] < highThreshold) &&
  21. (dis[(y + 1) * disdata.Stride + x - 1] < highThreshold) &&
  22. (dis[(y + 1) * disdata.Stride + x] < highThreshold) &&
  23. (dis[(y + 1) * disdata.Stride + x + 1] < highThreshold))
  24. {
  25. dis[y * disdata.Stride + x] = 0;
  26. }
  27. }
  28. }
  29. }
  30. }</span>

最后,效果图如下:

原图:

灰度图:

边缘图:

时间: 2024-10-20 07:13:01

Canny边缘检测原理及C#程序实现的相关文章

Log和Canny边缘检测(附Matlab程序)

  一. 实验目的 (1) 通过实验分析不同尺度下LOG和Canny边缘提取算子的性能. (2) 研究这两种边缘提取方法在不同参数下的边缘提取能力. (3) 使用不同的滤波尺度和添加噪声能量(噪声水平),通过与无噪声图像对比,选择最能说明自己结论的滤波尺度和噪声水平,并做出分析说明. 二. 实验原理 边缘的含义:在数字图像中,边缘是指图像局部变化最显著的部分,边缘主要存在于目标与目标,目标与背景之间,是图像局部特性的不连续性,如灰度的突变.纹理结构的突变.颜色的突变等.尽管图像的边缘点产生的原因

算法解剖系列(1)-Canny边缘检测原理

Canny边缘检测算子 基本原理 须满足条件:抑制噪声:精确定位边缘. 从数学上表达了三个准则[信噪比准则(低错误率).定位精度准则.单边缘响应准则],并寻找表达式的最佳解. 属于先平滑后求导的方法. 算法实现步骤 1.使用高斯滤波平滑图像 令f(x,y)表示数据(输入源数据),G(x,y)表示二维高斯函数(卷积操作数),fs(x,y)为卷积平滑后的图像. G(x,y)=12πσ2e?(x2+y2)2σ2 fs(x,y)=f(x,y)?G(x,y) Guess过程 用坐标点(x,y)表示一个3×

六 OpenCV图像处理4 Canny 边缘检测

1.Canny 边缘检测原理 步骤: ·1噪声去除: 由于边缘检测很容易受到噪声影响,所以第一步是使用 5x5 的高斯滤波器 去除噪声 ·2计算图像梯度: 对平滑后的图像使用 Sobel 算子计算水平方向和竖直方向的一阶导数(图 像梯度)(Gx 和 Gy) 根据得到的这两幅梯度图(Gx 和 Gy)找到边界的梯 度和方向 梯度的方向一般总是与边界垂直.梯度方向被归为四类:垂直,水平,和 两个对角线. ·3非极大值抑制 在获得梯度的方向和大小之后,应该对整幅图像做一个扫描,去除那些非 边界上的点.对

Canny边缘检测算法原理及其VC实现详解(一)

转自:http://blog.csdn.net/likezhaobin/article/details/6892176 图象的边缘是指图象局部区域亮度变化显著的部分,该区域的灰度剖面一般可以看作是一个阶跃,既从一个灰度值在很小的缓冲区域内急剧变化到另一个灰度相差较大的灰度值.图象的边缘部分集中了图象的大部分信息,图象边缘的确定与提取对于整个图象场景的识别与理解是非常重要的,同时也是图象分割所依赖的重要特征,边缘检测主要是图象的灰度变化的度量.检测和定位,自从1959提出边缘检测以来,经过五十多年

Canny边缘检测算法原理及其VC实现详解(二)

转自:http://blog.csdn.net/likezhaobin/article/details/6892629 3.  Canny算法的实现流程 由于本文主要目的在于学习和实现算法,而对于图像读取.视频获取等内容不进行阐述.因此选用OpenCV算法库作为其他功能的实现途径(关于OpenCV的使用,作者将另文表述).首先展现本文将要处理的彩色图片. 图2 待处理的图像 3.1 图像读取和灰度化 编程时采用上文所描述的第二种方法来实现图像的灰度化.其中ptr数组中保存的灰度化后的图像数据.具

[转载+原创]Emgu CV on C# (六) —— Emgu CV on Canny边缘检测

Canny边缘检测也是一种边缘检测方法,本文介绍了Canny边缘检测的函数及其使用方法,并利用emgucv方法将轮廓检测解算的结果与原文进行比较. 图像的边缘检测的原理是检测出图像中所有灰度值变化较大的点,而且这些点连接起来就构成了若干线条,这些线条就可以称为图像的边缘.Canny边缘检测算子是John F. Canny于 1986 年开发出来的一个多级边缘检测算法. Canny 边缘检测的数学原理和算法实现这里就不再了,有兴趣的读者可以查阅专业书籍. 一.概述(若果不想看,可以略过.转自:<C

OpenCV2马拉松第17圈——边缘检测(Canny边缘检测)

计算机视觉讨论群162501053 转载请注明:http://blog.csdn.net/abcd1992719g 收入囊中 利用OpenCV Canny函数进行边缘检测 掌握Canny算法基本理论 分享Java的实现 葵花宝典 在此之前,我们先阐述一下canny检测的算法.总共分为4部分. (1)处理噪声 一般用高斯滤波.OpenCV使用如下核 (2)计算梯度幅值 先用如下Sobel算子计算出水平和竖直梯度 我在OpenCV2马拉松第14圈--边缘检测(Sobel,prewitt,robert

Canny边缘检测算法

作为对比,先看一下Sobel的原理: Sobel的原理: 索贝尔算子(Sobeloperator)是图像处理中的算子之一,主要用作边缘检测.在技术上,它是一离散性差分算子,用来运算图像亮度函数的梯度之近似值.在图像的任何一点使用此算子,将会产生对应的梯度矢量或是其法矢量. 该算子包含两组3x3的矩阵,分别为横向及纵向,将之与图像作平面卷积,即可分别得出横向及纵向的亮度差分近似值.如果以A代表原始图像,Gx及Gy分别代表经横向及纵向边缘检测的图像 Canny边缘检测算子的matlab实现 在以上例

openCV实例:Canny边缘检测

http://blog.sina.com.cn/s/blog_737adf530100z0jk.html 在第一次使用openCV程序成功对图像进行打开后,现在开始试验第二个例程试验:Canny边缘检测 这里算法原理和具体编程语句都先不管,因为作为一名新手(反正我是这么感觉的),拿些现成的程序跑出效果才是让人很有feel的. 先贴下原代码: #include "cv.hpp"#include "cxcore.hpp"#include "opencv2\hi