0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

图像锐化的Sobel、Laplacian算子基础知识介绍

OSC开源社区 来源:OSCHINA 社区 2023-09-13 09:52 次阅读

一.Sobel 算子

Sobel 算子是一种用于边缘检测的离散微分算子,它结合了高斯平滑和微分求导。该算子用于计算图像明暗程度近似值,根据图像边缘旁边明暗程度把该区域内超过某个数的特定点记为边缘。Sobel 算子在 Prewitt 算子的基础上增加了权重的概念,认为相邻点的距离远近对当前像素点的影响是不同的,距离越近的像素点对应当前像素的影响越大,从而实现图像锐化并突出边缘轮廓 [1-4]。

Sobel 算子的边缘定位更准确,常用于噪声较多、灰度渐变的图像。其算法模板如公式(1)所示,其中 dx 表示水平方向,dy 表示垂直方向 [3]。

839f5a18-5160-11ee-a25d-92fbcf53809c.png

其像素计算公式如下:

83f27784-5160-11ee-a25d-92fbcf53809c.png

Sobel 算子像素的最终计算公式如下:

845352e8-5160-11ee-a25d-92fbcf53809c.png

Sobel 算子根据像素点上下、左右邻点灰度加权差,在边缘处达到极值这一现象检测边缘。对噪声具有平滑作用,提供较为精确的边缘方向信息。因为 Sobel 算子结合了高斯平滑和微分求导(分化),因此结果会具有更多的抗噪性,当对精度要求不是很高时,Sobel 算子是一种较为常用的边缘检测方法。

Python 和 OpenCV 将 Sobel 算子封装在 Sobel () 函数中,其函数原型如下所示:

dst = Sobel(src, ddepth, dx, dy[, dst[, ksize[, scale[, delta[, borderType]]]]])

– src 表示输入图像

– dst 表示输出的边缘图,其大小和通道数与输入图像相同

– ddepth 表示目标图像所需的深度,针对不同的输入图像,输出目标图像有不同的深度

– dx 表示 x 方向上的差分阶数,取值 1 或 0

– dy 表示 y 方向上的差分阶数,取值 1 或 0

– ksize 表示 Sobel 算子的大小,其值必须是正数和奇数

– scale 表示缩放导数的比例常数,默认情况下没有伸缩系数

– delta 表示将结果存入目标图像之前,添加到结果中的可选增量值

– borderType 表示边框模式,更多详细信息查阅 BorderTypes

注意,在进行 Sobel 算子处理之后,还需要调用 convertScaleAbs () 函数计算绝对值,并将图像转换为 8 位图进行显示。其算法原型如下:

dst = convertScaleAbs(src[, dst[, alpha[, beta]]])

– src 表示原数组

– dst 表示输出数组,深度为 8 位

– alpha 表示比例因子

– beta 表示原数组元素按比例缩放后添加的值

Sobel 算子的实现代码如下所示。

# -*- coding: utf-8 -*-

# By:Eastmount

import cv2

import numpy as np

import matplotlib.pyplot as plt



#读取图像

img = cv2.imread('luo.png')

lenna_img = cv2.cvtColor(img,cv2.COLOR_BGR2RGB)

#灰度化处理图像

grayImage = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)



#Sobel算子

x = cv2.Sobel(grayImage, cv2.CV_16S, 1, 0) #对x求一阶导

y = cv2.Sobel(grayImage, cv2.CV_16S, 0, 1) #对y求一阶导

absX = cv2.convertScaleAbs(x)

absY = cv2.convertScaleAbs(y)

Sobel = cv2.addWeighted(absX, 0.5, absY, 0.5, 0)

#用来正常显示中文标签

plt.rcParams['font.sans-serif']=['SimHei']

#显示图形

titles = ['原始图像', 'Sobel算子']

images = [lenna_img, Sobel]

for i in range(2):

plt.subplot(1,2,i+1), plt.imshow(images[i], 'gray')

plt.title(titles[i])

plt.xticks([]),plt.yticks([])

plt.show()
其运行结果如图 1 所示:

84a31562-5160-11ee-a25d-92fbcf53809c.png

二.Laplacian 算子

拉普拉斯(Laplacian)算子是 n 维欧几里德空间中的一个二阶微分算子,常用于图像增强领域和边缘提取。它通过灰度差分计算邻域内的像素,基本流程是:

判断图像中心像素灰度值与它周围其他像素的灰度值;

如果中心像素的灰度更高,则提升中心像素的灰度;

反之降低中心像素的灰度,从而实现图像锐化操作。

在算法实现过程中,Laplacian 算子通过对邻域中心像素的四方向或八方向求梯度,再将梯度相加起来判断中心像素灰度与邻域内其他像素灰度的关系,最后通过梯度运算的结果对像素灰度进行调整 [2]。 一个连续的二元函数 f (x,y),其拉普拉斯运算定义为: 85147658-5160-11ee-a25d-92fbcf53809c.png Laplacian 算子分为四邻域和八邻域,四邻域是对邻域中心像素的四方向求梯度,八邻域是对八方向求梯度。其中,四邻域模板如公式(5)所示: 8565d7d2-5160-11ee-a25d-92fbcf53809c.png 其像素的计算公式可以简化为: 85d7d670-5160-11ee-a25d-92fbcf53809c.png 通过模板可以发现,当邻域内像素灰度相同时,模板的卷积运算结果为 0;当中心像素灰度高于邻域内其他像素的平均灰度时,模板的卷积运算结果为正数;当中心像素的灰度低于邻域内其他像素的平均灰度时,模板的卷积为负数。对卷积运算的结果用适当的衰弱因子处理并加在原中心像素上,就可以实现图像的锐化处理。 Laplacian 算子的八邻域模板如下: 862ff2a6-5160-11ee-a25d-92fbcf53809c.png 其像素的计算公式可以简化为:

86850b6a-5160-11ee-a25d-92fbcf53809c.png

Python 和 OpenCV 将 Laplacian 算子封装在 Laplacian () 函数中,其函数原型如下所示:

dst = Laplacian(src, ddepth[, dst[, ksize[, scale[, delta[, borderType]]]]])

– src 表示输入图像

– dst 表示输出的边缘图,其大小和通道数与输入图像相同

– ddepth 表示目标图像所需的深度

– ksize 表示用于计算二阶导数的滤波器的孔径大小,其值必须是正数和奇数,且默认值为 1,更多详细信息查阅 getDerivKernels

– scale 表示计算拉普拉斯算子值的可选比例因子。默认值为 1,更多详细信息查阅 getDerivKernels

– delta 表示将结果存入目标图像之前,添加到结果中的可选增量值,默认值为 0

– borderType 表示边框模式,更多详细信息查阅 BorderTypes

注意,Laplacian 算子其实主要是利用 Sobel 算子的运算,通过加上 Sobel 算子运算出的图像 x 方向和 y 方向上的导数,得到输入图像的图像锐化结果。 同时,在进行 Laplacian 算子处理之后,还需要调用 convertScaleAbs () 函数计算绝对值,并将图像转换为 8 位图进行显示。其算法原型如下:

dst = convertScaleAbs(src[, dst[, alpha[, beta]]])

– src 表示原数组

– dst 表示输出数组,深度为 8 位

– alpha 表示比例因子

– beta 表示原数组元素按比例缩放后添加的值

当 ksize=1 时,Laplacian () 函数采用 3×3 的孔径(四邻域模板)进行变换处理。下面的代码是采用 ksize=3 的 Laplacian 算子进行图像锐化处理,其代码如下:

# -*- coding: utf-8 -*-

# By:Eastmount

import cv2

import numpy as np

import matplotlib.pyplot as plt



#读取图像

img = cv2.imread('luo.png')

lenna_img = cv2.cvtColor(img,cv2.COLOR_BGR2RGB)

#灰度化处理图像

grayImage = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)



#拉普拉斯算法

dst = cv2.Laplacian(grayImage, cv2.CV_16S, ksize = 3)

Laplacian = cv2.convertScaleAbs(dst)

#用来正常显示中文标签

plt.rcParams['font.sans-serif']=['SimHei']

#显示图形

titles = ['原始图像', 'Laplacian算子']

images = [lenna_img, Laplacian]

for i in range(2):

plt.subplot(1,2,i+1), plt.imshow(images[i], 'gray')

plt.title(titles[i])

plt.xticks([]),plt.yticks([])

plt.show()
其运行结果如图 2 所示:

86dda61c-5160-11ee-a25d-92fbcf53809c.png

边缘检测算法主要是基于图像强度的一阶和二阶导数,但导数通常对噪声很敏感,因此需要采用滤波器来过滤噪声,并调用图像增强或阈值化算法进行处理,最后再进行边缘检测。下面是采用高斯滤波去噪和阈值化处理之后,再进行边缘检测的过程,并对比了四种常见的边缘提取算法。

# -*- coding: utf-8 -*-

# By:Eastmount

import cv2

import numpy as np

import matplotlib.pyplot as plt

#读取图像

img = cv2.imread('luo.png')

lenna_img = cv2.cvtColor(img, cv2.COLOR_BGR2RGB)

#灰度化处理图像

grayImage = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)

#高斯滤波

gaussianBlur = cv2.GaussianBlur(grayImage, (3,3), 0)

#阈值处理

ret, binary = cv2.threshold(gaussianBlur, 127, 255, cv2.THRESH_BINARY)

#Roberts算子

kernelx = np.array([[-1,0],[0,1]], dtype=int)

kernely = np.array([[0,-1],[1,0]], dtype=int)

x = cv2.filter2D(binary, cv2.CV_16S, kernelx)

y = cv2.filter2D(binary, cv2.CV_16S, kernely)

absX = cv2.convertScaleAbs(x)

absY = cv2.convertScaleAbs(y)

Roberts = cv2.addWeighted(absX, 0.5, absY, 0.5, 0)

#Prewitt算子

kernelx = np.array([[1,1,1],[0,0,0],[-1,-1,-1]], dtype=int)

kernely = np.array([[-1,0,1],[-1,0,1],[-1,0,1]], dtype=int)

x = cv2.filter2D(binary, cv2.CV_16S, kernelx)

y = cv2.filter2D(binary, cv2.CV_16S, kernely)

absX = cv2.convertScaleAbs(x)

absY = cv2.convertScaleAbs(y)

Prewitt = cv2.addWeighted(absX,0.5,absY,0.5,0)

#Sobel算子

x = cv2.Sobel(binary, cv2.CV_16S, 1, 0)

y = cv2.Sobel(binary, cv2.CV_16S, 0, 1)

absX = cv2.convertScaleAbs(x)

absY = cv2.convertScaleAbs(y)

Sobel = cv2.addWeighted(absX, 0.5, absY, 0.5, 0)

#拉普拉斯算法

dst = cv2.Laplacian(binary, cv2.CV_16S, ksize = 3)

Laplacian = cv2.convertScaleAbs(dst)

#效果图

titles = ['Source Image', 'Binary Image', 'Roberts Image',

'Prewitt Image','Sobel Image', 'Laplacian Image']

images = [lenna_img, binary, Roberts, Prewitt, Sobel, Laplacian]

for i in np.arange(6):

plt.subplot(2,3,i+1),plt.imshow(images[i],'gray')

plt.title(titles[i])

plt.xticks([]),plt.yticks([])

plt.show()
输出结果如图 3 所示。其中,Laplacian 算子对噪声比较敏感,由于其算法可能会出现双像素边界,常用来判断边缘像素位于图像的明区或暗区,很少用于边缘检测;Robert 算子对陡峭的低噪声图像效果较好,尤其是边缘正负 45 度较多的图像,但定位准确率较差;Prewitt 算子对灰度渐变的图像边缘提取效果较好,而没有考虑相邻点的距离远近对当前像素点的影响;Sobel 算子考虑了综合因素,对噪声较多的图像处理效果更好。

8757a228-5160-11ee-a25d-92fbcf53809c.png

三、总结

本文主要介绍图像锐化和边缘检测知识,详细讲解了 Sobel 算子和 Laplacian 算子,并通过小珞珞图像进行边缘轮廓提取。图像锐化和边缘提取技术可以消除图像中的噪声,提取图像信息中用来表征图像的一些变量,为图像识别提供基础。







审核编辑:刘清

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 滤波器
    +关注

    关注

    158

    文章

    7419

    浏览量

    175577
  • 边缘检测
    +关注

    关注

    0

    文章

    91

    浏览量

    18141
  • 拉普拉斯
    +关注

    关注

    0

    文章

    32

    浏览量

    9604
  • OpenCV
    +关注

    关注

    29

    文章

    614

    浏览量

    40909
  • python
    +关注

    关注

    53

    文章

    4701

    浏览量

    83700

原文标题:详解图像锐化的Sobel、Laplacian算子

文章出处:【微信号:OSC开源社区,微信公众号:OSC开源社区】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    DSP c6722 图像锐化

    求LOG算子,canny算子 ,Roberts算子,梯度算子锐化程序。跪谢
    发表于 05-12 10:14

    基于FPGA的Sobel边缘检测的实现

    我们在此基础上修改,从而实现,基于FPGA的动态图片的Sobel边缘检测、中值滤波、Canny算子边缘检测、腐蚀和膨胀等。那么这篇文章我们将来实现基于FPGA的Sobel边缘检测。图像
    发表于 08-29 15:41

    基于FPGA的图像拉普拉斯锐化处理

    时,此中心像素的灰度应被进一步提高,以此实现图像锐化处理。2.2拉普拉斯(laplace)算子最常用的无方向性的二阶差分算子,其模板有3*3、5*5和7*7等多种形式。。例如,以3*
    发表于 07-08 18:15

    源码交流=图像处理 实现图像去噪、滤波、锐化、边缘检测

    均衡化、锐化、边缘检测【处理效果】NO.1:原图NO.2:去噪之后的图像NO.3:锐化之后的图像NO.4:直方图均衡之后的图像NO.5:Pr
    发表于 04-01 19:03

    Labview图像处理——边缘检测

    ,而是用来判断边缘像素视为与图像的明区还是暗区。拉普拉斯高斯算子是一种二阶导数算子,将在边缘处产生一个陡峭的零交叉, Laplacian算子
    发表于 12-01 12:16

    边缘检测的几种微分算子

    一、边缘检测边缘检测的几种微分算子:一阶微分算子:Roberts、Sobel、Prewitt二阶微分算子Laplacian、Log/Mar
    发表于 07-26 08:29

    迅为iTOP-RK3568开发板Sobel 算子边缘检测

    对 number.png 图片进行读取; 第 3 行使用了 imshow()函数对原图像进行展示; 第 4 行使用 Sobel 算子进行边缘检测计算,数据类型设置为 cv2.CV_64F,只算 x 方向梯 度
    发表于 09-18 10:27

    基于改进的Laplacian算子图像边缘检测

    分析了图像边缘特性以及Laplacian算子检测图像边缘的基本原理!并对经典Laplacian算子
    发表于 05-17 10:46 29次下载
    基于改进的<b class='flag-5'>Laplacian</b><b class='flag-5'>算子</b><b class='flag-5'>图像</b>边缘检测

    基于拉普拉斯算法的图像锐化算法实现

    该文提出了一种基于拉普拉斯算法的图像锐化方法,并在DSP上实现其算法。首先研究拄普拉斯算子锐化图像的基本原理,并推导出
    发表于 10-12 16:22 79次下载
    基于拉普拉斯算法的<b class='flag-5'>图像</b><b class='flag-5'>锐化</b>算法实现

    Laplacian算子的FPGA实现方法

    拉普拉斯算子是一种重要的图像增强算子,它是一种各向同性滤波器,即滤波器的响应与滤波器作用图像的突变方向无关,而且实现简单,被广泛用于图像
    的头像 发表于 06-16 17:47 3070次阅读
    <b class='flag-5'>Laplacian</b><b class='flag-5'>算子</b>的FPGA实现方法

    FPGA图像处理的Sobel边缘检测

    Sobel边缘检测 Sobel边缘检测原理教材网上一大堆,核心为卷积处理。 Sobel卷积因子为: 该算子包含两组3x3的矩阵,分别为横向及纵向,将之与
    的头像 发表于 03-22 09:45 2761次阅读
    FPGA<b class='flag-5'>图像</b>处理的<b class='flag-5'>Sobel</b>边缘检测

    浅述Sobel算子在HLS上的实现教程

    Sobel 原理介绍 索贝尔算子Sobel operator)主要用作边缘检测,在技术上,它是一离散性差分算子,用来运算
    的头像 发表于 07-23 14:53 2032次阅读
    浅述<b class='flag-5'>Sobel</b><b class='flag-5'>算子</b>在HLS上的实现教程

    Sobel边缘检测与锐化的实现

    效果的影响,需要利用图像锐化技术,使图像的边缘变得清晰。图像锐化处理主要有两个目的:一是与图像
    的头像 发表于 03-21 13:17 2538次阅读

    Laplacian算子的硬件实现及结果

    使用Laplacian算子滤波是将模板与图像做卷积运算,然后将得到的结果取绝对值后,再进行防治溢出(灰度值大于255)处理。所以在用硬件实现Laplacian
    发表于 07-21 09:27 819次阅读

    Sobel算子原理介绍与实现方法

    索贝尔算子Sobel operator)主要用作边缘检测,在技术上,它是一离散性差分算子,用来运算图像亮度函数的灰度之近似值。在图像的任何
    的头像 发表于 07-21 17:27 1.1w次阅读