继续浏览精彩内容
慕课网APP
程序员的梦工厂
打开
继续
感谢您的支持,我会继续努力的
赞赏金额会直接到老师账户
将二维码发送给自己后长按识别
微信支付
支付宝支付

图像的边缘检测

青春有我
关注TA
已关注
手记 1072
粉丝 205
获赞 1007

最近整理一些学习笔记,因为觉得自己基础不扎实,很多算法能用却不知其根源,因此需要不断去回顾和归纳。这篇先写写图像的边缘检测。

边缘检测就是一个寻找边缘像素的过程,这些边缘一般就是局部像素变化比较显著的一些点,主要存在于目标和目标以及前景背景之间。找到这些边缘,对于图像分割与目标识别是很有帮助的。

常用的边缘检测算子有很多,这里以sobel、拉普拉斯算子和canny算子为例,做一些比较。

Sobel算子

Sobel 算子结合了高斯平滑和微分求导。它是一阶导数的边缘检测算子,使用卷积核与图像中的每个像素点做卷积和运算,然后采用合适的阈值提取边缘。Soble算子有两个卷积核,分别对应的是x与y两个方向。

计算过程

1.分别在x和y两个方向求导。

2.在图像的每一点,结合以上两个结果求出近似梯度:

cv2.Sobel(src, ddepth, dx, dy[, dst[, ksize[, scale[, delta[, borderType]]]]])
  • src是需要处理的图像;

  • ddepth是图像的深度,-1表示采用的是与原图像相同的深度。目标图像的深度必须大于等于原图像的深度;这里使用了cv2.CV_16S 即16位有符号的数据类型,以防止截断。 - dx和dy表示的是求导的阶数,0表示这个方向上没有求导,一般为0、1、2。

  • ksize是Sobel算子的大小,必须为1、3、5、7。

Laplace算子

拉普拉斯算子是一种二阶导数算子。在图像中的边缘区域,像素值会发生比较大的变化,对这些像素求导,会看到极值出现,在这些极值位置,其二阶导数为0,所以也可以用二阶导数来检测图像边缘。

Laplacian算子的定义:

webp

laplace.gif

cv2.Laplacian(src, ddepth[, dst[, ksize[, scale[, delta[, borderType]]]]])
  • src是需要处理的图像;

  • ddepth是图像的深度,解释同上文sobel。

  • ksize是算子的大小,必须为1、3、5、7。默认为1。

Canny

Canny算子的基本思想是寻找梯度的局部最大值。算子是比较简单的,也很容易实现。

一般步骤:
1. 高斯滤波图像去噪。一般情况下,使用高斯平滑滤波器卷积降噪。
2.计算梯度幅值和方向。运用一对卷积阵列(分别作用于x和y方向)
3.非极大值抑制。这一步排除非边缘像素, 仅仅保留了一些细线条(候选边缘)。
4.需要两个阈值(高阈值和低阈值)检测和连接边缘:

如果某一像素位置的幅值超过高阈值, 这个像素被保留为边缘像素。如果小于低阈值, 该像素被排除。如果在两个阈值之间,要看这个像素的邻接像素中有没有超过高阈值的边缘像素。如果有,这像素就是边缘,否则就不是。

cv2.Canny(scr, threshold1, threshold2[, edges[, apertureSize[, L2gradient]]])

这里scr是需要处理的原图像,该图像必须为单通道的灰度图;   threshold1是阈值1;   threshold2是阈值2。

实例代码

下面用Python-OpenCV代码比较下上面几种算子的效果。

"""
Created on Thu Oct 25 11:25:15 2018
@author: 晚晴风
"""import cv2

img = cv2.imread("pascal_19.jpg")#cannygray = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY)
blur = cv2.GaussianBlur(gray,(3,3),0)#高斯滤波器对图像降噪canny = cv2.Canny(blur,50,150)

cv2.imshow('canny',canny)#sobelgrad_x = cv2.Sobel(img,cv2.CV_16S,1,0,ksize = 3)
grad_y = cv2.Sobel(img,cv2.CV_16S,0,1,ksize = 3)#第二个参数为图像的深度u_x = cv2.convertScaleAbs(grad_x) #转化成为8位图形CV_8U进行显示u_y = cv2.convertScaleAbs(grad_y)
sobel_xy = cv2.addWeighted(u_x,0.5,u_y,0.5,0)#两方向加权求近似梯度cv2.imshow('sobelX',u_x)
cv2.imshow('sobelY',u_y)
cv2.imshow('sobelXY',sobel_xy)#Laplacelap = cv2.Laplacian(img,cv2.CV_16S,ksize = 3)
laplacian = cv2.convertScaleAbs(lap)

cv2.imshow('laplacian',laplacian)
cv2.waitKey(0)
cv2.destroyAllWindows()#释放窗口

运行之后就可以看到三种算子的边缘检测结果:

原图:

webp

pascal_19.jpg

Canny算子:

webp

canny.jpg

Sobel算子:

webp

sobelXY.jpg

Laplace算子:

webp

laplacian.jpg

总结:
  • Sobel算子在边缘检测的同时尽量的削弱了噪声。比较容易实现,受噪声的影响力比较小。它对于像素位置的影响作了加权,因此效果更好、应用广泛。

  • Laplace算子是一种各向同性算子,在只关心边缘的位置而不考虑其周围的象素灰度差值时比较合适。Laplace算子对孤立象素的响应要比对边缘或线的响应要更强烈,因此只适用于无噪声图象。存在噪声情况下,使用Laplacian算子检测边缘之前需要先进行低通滤波。

  • Canny算子是目前理论上相对最完善的一种边缘检测算法。也存在不足之处: 为了得到较好的边缘检测结果,它通常需要使用较大的滤波尺度,这样容易丢失一些细节。算子的双阈值要人为的选取,这需要经验:)



作者:晚晴风_
链接:https://www.jianshu.com/p/43557d1302d2


打开App,阅读手记
1人推荐
发表评论
随时随地看视频慕课网APP