Canny算法是边缘检测的一个经典算法,比单纯用一些微分算子来检测的效果要好很多,其优势有以下几点:

  1. 边缘误检与漏检率低。
  2. 边缘定位准确,且边界较细。
  3. 自带一定的滤噪功能,或者说,对噪声的敏感度要比单纯算子低。
  4. 具有多个可调整参数,可影响算法的时间与时效。

  但是Canny相比单纯算子来说计算量偏大,下面简单介绍算法的过程。

图像去噪: 

    这一步不是必须的,一般噪声少的图,让Canny自己应付就行。若噪声较多,一般采用高斯滤波。滤波后,噪声灰度下降,对边缘的影响

  小于噪点。

获取梯度强度与方向: 

    用一阶微分算子获取梯度强度和方向,如sobel算子,强度与方向公式如下:

    

    要用两个矩阵分别存储强度和方向,其中方向公式需要根据具体算子情况更改正负号,最终使上下左右、45度方向可以区分开来。例如下图所示,

    这里将上下放在(60,90)和(-90,-60)区间,因为它们在非极大抑制时取的都是上下邻域;将左右放在(-30,30)区间,因为它们抑制时取的是

    左右邻域;将处于斜向上对角线的梯度角放在(30,60),它们在抑制时取45度的两个邻域;将斜向下对角线上的梯度角放在(-60,-30),它们在

    抑制时取-45度的两个邻域。

    

非极大抑制:

    该步骤目的是删除非边缘像素,主要做法是对每个像素点,与它梯度方向的相邻两像素作灰度比较(比如左右两个像素),如果该像素点

  灰度比它两个相邻像素都大,则保留其灰度值,否则抑制(赋值0)。

双阈值处理:

    设定一个高阈值和低阈值,对灰度大于高阈值的像素点,确认为强边缘,赋值255,低于低阈值的点赋值0,介于之间的点保留其灰度值,作为弱边缘。

滞后边界追踪:

    这一步主要是处理弱边缘,总的思路是找到每一个弱边缘的连通域,判断该连通域与强边缘有无相邻,如果存在至少一个像素与强边缘相邻,则将该连

  通域都作为强边缘(赋值255),如果没有一个像素与强边缘相邻,则将整个连通域视为噪声(赋值0)。

    所以,关键在于找到连通域,通常有DFS与BFS两种算法可以处理连通域问题,这里采用BFS算法,思路如下:

    1.扫描整个图像,判断每个像素是否为弱边缘,如果是,进入步骤2(查找连通域)。

    2.用connect数组保存组成连通域的所有点;用weak数组保存待检查八领域的弱点;用checked数组标记已被扫描过的弱点(也就是连通域里的点),标

       记1表示已扫描;用变量real_edge记录是否有检测到强点。将刚才判定的弱边缘点压入connect、weak,checked相应位置标记为1。进入步骤3。

    3.创建new_weak用来保存新的待检测八领域的弱点。依次对weak中每个待检测点,遍历其八领域,找到所有未被checked标记的新弱点,压入new_weak

       、connect,checked相应位置标记为1,同时检查八领域内是否有强边缘,有则标记real_edge=1。对weak所有待检测点检查完后,如果new_weak不为

      空,则将new_weak赋给weak,重复步骤3;如果new_weak为空,则表示一个连通域寻找完毕,进入步骤4。

    4.如果real_edge=1,则将连通域每个点的灰度赋值255,否则赋值0;弹出connect、weak、checked中所有点。

    5.重复步骤1-4,直道图像扫描完毕。

    

  经过以上这些步骤,Canny算法就已经实现了。我们要做的就是根据图像调整双阈值和滤波强度。下面是Canny的一个处理实例,其中给出了单纯sobel检测和

Canny检测的效果。通过下图可以知道,Canny检测的边缘要薄的多,细节处理更好,噪声也更少。

 

  

  以下是matlab代码实现:

  

%canny前先高斯滤波
function edge=canny(gaussianimg,lthres,hthres)
sobel_operator_x=[-,,;
-,,;
-,,];
sobel_operator_y=[-,-,-;
,,;
,,];
%梯度强度
diff_x=filter2(sobel_operator_x,gaussianimg);
diff_y=filter2(sobel_operator_y,gaussianimg);
diff=uint8(sqrt((diff_x.^+diff_y.^)/));
%梯度方向
[sizex,sizey]=size(gaussianimg);
angle=zeros(sizex,sizey);
edge=uint8(zeros(sizex,sizey));
for i=:sizex
for j=:sizey
angle(i,j)=atan(-diff_y(i,j)/diff_x(i,j))/pi*;
end
end
%非极大抑制,排除非边缘像素
for i=:sizex-
for j=:sizey-
if (angle(i,j)>||angle(i,j)<-)
break;
elseif angle(i,j)>= || angle(i,j)<=-
if (diff(i,j)>diff(i-,j)&&diff(i,j)>=diff(i+,j))
edge(i,j)=uint8(diff(i,j));
end
elseif (angle(i,j)<=-)
if (diff(i,j)>diff(i-,j-)&&diff(i,j)>=diff(i+,j+))
edge(i,j)=uint8(diff(i,j));
end
elseif angle(i,j)>=
if (diff(i,j)>diff(i-,j+)&&diff(i,j)>=diff(i+,j-))
edge(i,j)=uint8(diff(i,j));
end
elseif angle(i,j)<||angle(i,j)>-
if (diff(i,j)>diff(i,j-)&&diff(i,j)>=diff(i,j+))
edge(i,j)=uint8(diff(i,j));
end
end
end
end %双阈值
for i=:sizex
for j=:sizey
if (edge(i,j)>=hthres)
edge(i,j)=; %一定为边缘
elseif (edge(i,j)<=lthres)
edge(i,j)=; %一定为非边缘
end
end
end %候选边缘,与已确定边缘相连才认为是边缘
for i=:sizex-
for j=:sizey-
if (edge(i,j)>&&edge(i,j)<)
real_edge=;%边缘真假标志
checked=zeros(sizex,sizey);%标记已经扫描过的弱点
weak=zeros(,);%存储需要查看八领域的弱点
connect=zeros(,);%存储一条联通的所有弱点
weak_length=;
connect_length=;
%压入第一个弱边缘点
weak(weak_length,:)=[i,j];
connect(connect_length,:)=[i,j];
checked(i,j)=;
while(weak_length>)
new_weak=zeros(,);
new_weak_length=;
for k=:weak_length
%搜索当前弱点的八领域
x=weak(k,);y=weak(k,);
if (x>=&&x<=sizex-&&y>=&&y<=sizey-)
for m=x-:x+
for n=y-:y+
if edge(m,n)>&&edge(m,n)<&&checked(m,n)==%领域有弱点且未被扫描过
new_weak_length=new_weak_length+;
connect_length=connect_length+;
new_weak(weak_length,:)=[m,n];%压入新弱点集合
connect(connect_length,:)=[m,n]; %压入连通域
checked(m,n)=; %标记已扫描
elseif edge(m,n)==
real_edge=; %边缘为真,等待连通域全部被识别
end
end
end
end
end
weak_length=new_weak_length;%当前深度的弱点集合全部检查过八领域,开始检查新一深度的弱点集合
weak=new_weak;%如果新集合没有弱点,则跳出while
end
%一个连通域已在connect里形成
for z=:connect_length
edge(connect(z,),connect(z,))=uint8((* real_edge));
end end
end
end
end

  

Canny算法检测边缘的更多相关文章

  1. Kosaraju 算法检测有向图的强连通性

    给定一个有向图 G = (V, E) ,对于任意一对顶点 u 和 v,有 u --> v 和 v --> u,亦即,顶点 u 和 v 是互相可达的,则说明该图 G 是强连通的(Strong ...

  2. opencv目标检测之canny算法

    canny canny的目标有3个 低错误率 检测出的边缘都是真正的边缘 定位良好 边缘上的像素点与真正的边缘上的像素点距离应该最小 最小响应 边缘只能标识一次,噪声不应该标注为边缘 canny分几步 ...

  3. [CLPR] 定位算法探幽 - 边缘和形态学

    一. 引言 如何从一副图片中找到车牌? 这是机器视觉的一个应用. 理所当然地, 思考的角度是从车牌本身的信息入手, 为了讨论方便, 下面均以长窄型蓝白车牌为例. 下图就是这样一张车牌的基本信息. 一眼 ...

  4. [算法]检测空间三角形相交算法(Devillers & Guigue算法)

    #pragma once //GYDevillersTriangle.h /* 快速检测空间三角形相交算法的代码实现(Devillers & Guigue算法) 博客原地址:http://bl ...

  5. Iconfinder 如何杜绝盗版,哈希算法检测图像重复

    原地址:http://blog.jobbole.com/65914/ 本文由 伯乐在线 - 小鱼 翻译自 Silviu Tantos.欢迎加入技术翻译小组.转载请参见文章末尾处的要求. [伯乐在线导读 ...

  6. miller_rabin算法检测生成大素数的RSA算法实现

      import math from functools import reduce #用于合并字符 from os import urandom #系统随机的字符 import binascii # ...

  7. Python3实现Two-Pass算法检测区域连通性

    技术背景 连通性检测是图论中常常遇到的一个问题,我们可以用五子棋的思路来理解这个问题五子棋中,横.竖.斜相邻的两个棋子,被认为是相连接的,而一样的道理,在一个二维的图中,只要在横.竖.斜三个方向中的一 ...

  8. Halcon使用MeasurePos来实现检测边缘点

    (1)为了提高性能,测量句柄只需要初始化一次: 参数:测量矩形的中心点行坐标,测量矩形中心的列坐标,测量矩形的角度,测量矩形的宽,测量矩形的高,待处理图像的宽,待处理图像的高,使用的算法,输出测量句柄 ...

  9. 【OpenCV, C++】实现向下光栅追踪检测边缘

    设计函数如下: 其中 void gratingdetect(Mat &graysrc, Mat &graydst, int high, int low); 参数列表中,第一项是输入的灰 ...

随机推荐

  1. WebJars简介 —— 前端资源的jar包形式(以后接触到再深入总结)

    对于日常的web开发而言,像css.js.images.font等静态资源文件管理是非常的混乱的.比如jQuery.Bootstrap.Vue.js等,可能每个框架使用的版本都不一样.一不注意就会出现 ...

  2. C++面向对象程序设计学习笔记(1)

    基本概念 对象: 面向对象程序设计中,对象是描述其属性的数据以及对这些数据施加的一组操作封装在一起构成的统一体,每个对象都是由数据和操作代码两部分构成的. 类: 面向对象程序设计中,类是具有相同的数据 ...

  3. appium--多进程启动多设备

    前戏 在前面我们都是使用一个机器进行测试,在做app自动化的时候,我们要测不同的机型,也就是兼容性测试,如果一台一台设备去执行,那就显的太麻烦了.所以经常需要我们启动多个设备,同时跑自动化测试用例,要 ...

  4. 题解 P2668 【斗地主】

    dfs+简易剪枝+简易a* 思路: dfs+简易剪枝+简易a(我也不知道算不算a): dfs参数记录层数 按消耗牌多少的贪心顺序搜索 有几种情况可以不用搜索(但我还是搜索了) 可以用a*估算出来 最后 ...

  5. Linux性能优化实战学习笔记:第十三讲

    问题1:性能工具版本太低,导致指标不全 解决方案1: 这是使用 CentOS 的同学普遍碰到的问题.在文章中,我的pidstat 输出里有一个 %wait 指标,代表进程等待 CPU 的时间百分比, ...

  6. Leetcode 第135场周赛解题报告

    这周比赛的题目很有特点.几道题都需要找到一定的技巧才能巧妙解决,和以往靠数据结构的题目不太一样. 就是如果懂原理,代码会很简单,如果暴力做,也能做出来,但是十分容易出错. 第四题还挺难想的,想了好久才 ...

  7. Axure入门

    一.Axure介绍 1.1 原型和Axure是什么? 原型是什么? 简单的说就是产品设计成形之前的一个简单框架,对网站来讲,就是将页面模块.元素进行粗放式的排版和布局,深入一些,还会加入一些交互性的元 ...

  8. 向github项目push代码后,Jenkins实现其自动构建

    配置Jenkins(添加Github服务器) 1.进入[系统管理] --> [系统设置] ,找到[Github] 2.添加Github服务器 这里需要github提供一个密钥文本,我们去gith ...

  9. Can't locate Math/Round.pm in @INC

    遭遇报错: Can't locate Math/Round.pm in @INC 经过亲自测试,下面的命令解决了我的问题. yum install perl-Math-Round 参考资料 ===== ...

  10. java 中遍历Map的几种方法

    方法分为两类: 一类是基于map的Entry:map.entrySet(); 一类是基于map的key:map.keySet() 而每一类都有两种遍历方式: a.利用迭代器 iterator: b.利 ...