OpenCV-Python(19):Canny边缘检测

目录

 学习目标

Canny 边缘检测原理

1.噪声抑制(噪声去除)

2.梯度计算

3.非极大值抑制

4.双阈值检测(滞后阈值)

5.边缘连接

Canny 边缘检测步骤

Canny 边缘检测的OpenCV实现 

不同阈值的边缘检测效果


学习目标

  • 了解Canny边缘检测的概念
  • 学习掌握函数cv2.Canny()的用法

Canny 边缘检测原理

        Canny 边缘检测是一种非常流行的边缘检测算法,是John F.Canny 在1986 年提出的。它是一个有很多步构成的算法,该算法的原理基于以下几个关键思想,我们接下来会分步介绍。

1.噪声抑制(噪声去除)

        由于边缘检测很容易受到噪声影响,在进行边缘检测之前,一般先使用高斯滤波器对图像进行平滑处理(通常使用5*5的核),以减少噪声的影响。高斯滤波器可以有效地模糊图像,同时保留边缘的细节。

2.梯度计算

        对平滑后的图像使用Sobel算子(Sobel算子是一种线性滤波器,可以检测图像中的边缘)计算图像在水平和垂直方向上的一阶导数(图像梯度),根据得到的两幅梯度图Gx和Gy找到边界的梯度和方向,公式如下:

梯度的方向一般总是与边界垂直。梯度方向归为四类:垂直、水平和两个对角线。 

3.非极大值抑制

        在获得梯度的方向和大小之后,应该对整幅图像做一个扫描,去除那些非边界上的点。对每一个像素进行检查,看这个点的梯度是不是周围具有相同梯度方向的点中最大的(选择梯度最大的像素作为边缘点,从而抑制非边缘点)。如下图所示:

4.双阈值检测(滞后阈值)

        为了确定哪些边界才是真正的边界,需要设定两个阈值minVal 和maxVal。当图像的灰度梯度高于maxVal 时,被认为为是真的边界,那些低于minVal 的边界界会被抛弃。如果介于两者之间的就要看这个点是否与某个被确定为真正的边界点相连,如果是就认为它也是边界点,如果不是就抛弃。如下图:

A 高于阈值maxVal 所以是真正的边界点,C 虽然低于maxVal 但由于高于minVal 并且与A 相连,所以也被认为为是真正的边界点。而B 就会抛弃,因为它不仅低于maxVal 而且不与真正的边界点相连。所以选择合适的maxVal和minVal 对于能否得到好的结果非常重要。在这一步一些小的噪声点也会被除去,因为我们假设边界都是一些比较长的线 段。

5.边缘连接

        通过迭代地访问弱边缘像素,并与其相邻的强边缘像素进行连接,最终确定真正的边缘。

Canny 边缘检测步骤

  1. 将输入图像转换为灰度图像,因为Canny边缘检测只适用于单通道灰度图像。
  2. 对灰度图像进行高斯滤波,以减少图像中的噪声。高斯滤波可以使用cv2.GaussianBlur()函数实现。
  3. 使用Sobel算子计算图像的梯度。Sobel算子可以分别计算图像在水平和垂直方向上的梯度。这可以通过cv2.Sobel()函数实现。
  4. 根据梯度幅值和方向计算边缘的强度和方向。幅值较大的像素被认为是边缘像素,而幅值较小的像素被认为是非边缘像素。
  5. 应用非极大值抑制,以消除边缘响应中的次要边缘。这可以通过比较每个像素的梯度方向与其相邻像素的梯度方向来实现。
  6. 应用双阈值来确定真正的边缘。将像素分为强边缘、弱边缘和非边缘三类。强边缘像素被认为是真正的边缘,弱边缘像素需要进一步确认,非边缘像素被排除。
  7. 使用连接弱边缘像素的方法来连接真正的边缘。这可以通过迭代地访问弱边缘像素并检查其相邻像素的强边缘状态来实现。
  8. 最后,得到的边缘图像包含了检测到的边缘。

注意:Canny边缘检测的结果通常是二值图像,其中边缘像素为白色,非边缘像素为黑色。

Canny 边缘检测的OpenCV实现 

        在OpenCV 中只需要一个函数cv2.Canny()就可以完成以上几步,该函数是OpenCV中用于执行Canny边缘检测的函数。其语法如下:

edges = cv2.Canny(image, threshold1, threshold2, apertureSize=None, L2gradient=None)

参数说明:

  • image:要进行边缘检测的输入图像。必须是单通道灰度图像
  • threshold1:第一个阈值,用于边缘链接。较低的阈值用于检测强边缘。
  • threshold2:第二个阈值,用于边缘链接。较高的阈值用于检测弱边缘,并将其链接到强边缘。
  • apertureSize:可选参数,Sobel算子的孔径大小。默认值为3。
  • L2gradient:可选参数,指定计算梯度幅值的方法。如果为True,则使用更精确的L2范数进行计算。默认值为False,使用以下方式计算:

cv2.Canny()函数将返回一个二值图像,其中包含检测到的边缘。

下面是一个使用cv2.Canny()函数进行Canny边缘检测的示例:

import cv2# 读取图像
img = cv2.imread('image.jpg', cv2.IMREAD_GRAYSCALE)# 进行Canny边缘检测
edges = cv2.Canny(img, 100, 200)# 显示结果图像
cv2.imshow('Canny Edge Detection', edges)
cv2.waitKey(0)
cv2.destroyAllWindows()

上述代码中的image.jpg是待处理的图像文件路径,你需要将其替换为你自己的图像文件路径。该示例将使用阈值100200进行Canny边缘检测,并显示结果图像。你可以根据需要调整阈值以获取更好的边缘检测结果。

不同阈值的边缘检测效果

下面是使用滑动条进行Canny边缘检测的Python代码示例,可以通过调节滑动条来设置阈值minVal 和maxVal 的大小来进行Canny 边界检测:

import cv2
import numpy as npdef nothing(x):pass# 创建窗口和滑动条
cv2.namedWindow('Canny Edge Detection')
cv2.createTrackbar('minVal', 'Canny Edge Detection', 0, 255, nothing)
cv2.createTrackbar('maxVal', 'Canny Edge Detection', 0, 255, nothing)# 读取图像
img = cv2.imread('lena.jpg', cv2.IMREAD_GRAYSCALE)while True:# 获取滑动条的值minVal = cv2.getTrackbarPos('minVal', 'Canny Edge Detection')maxVal = cv2.getTrackbarPos('maxVal', 'Canny Edge Detection')# 使用滑动条的值进行Canny边缘检测edges = cv2.Canny(img, minVal, maxVal)# 显示结果图像cv2.imshow('Canny Edge Detection', edges)# 按下ESC键退出循环if cv2.waitKey(1) == 27:break

        在运行代码时,会弹出一个名为"Canny Edge Detection"的窗口,其中包含两个滑动条用于调节Canny边缘检测的阈值。通过调整滑动条的值,你可以实时查看Canny边缘检测的结果,这样你就会理理解阈值的重要性了,按下ESC键可退出循环。以下是运行效果图:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/294529.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

法线贴图实现地形模型皱褶、凹凸不平的纹理效果

在线工具推荐: 3D数字孪生场景编辑器 - GLTF/GLB材质纹理编辑器 - 3D模型在线转换 - Three.js AI自动纹理开发包 - YOLO 虚幻合成数据生成器 - 三维模型预览图生成器 - 3D模型语义搜索引擎 法线贴图在3D建模中扮演着重要的角色,它通过模拟表面的微…

ERP系统成本管理:提升企业成本管控

一、ERP系统成本管理概述 ERP系统成本管理是指通过ERP系统对企业成本进行全面、实时、准确的管理和控制。它涵盖了成本核算、成本分析、成本控制等多个方面,为企业提供了一个统一、集成的成本管理平台。 二、ERP系统成本管理的优势 统一成本管理流程:E…

50 个具有挑战性的概率问题 [01/50]:袜子抽屉

一、说明 我最近对与概率有关的问题产生了兴趣。我偶然读到了弗雷德里克莫斯特勒(Frederick Mosteller)的《概率论中的五十个具有挑战性的问题与解决方案》(Fifty Challenge Problems in Probability with Solutions)一书。我认为…

JavaScript原型,原型链 ? 有什么特点?

一、原型 JavaScript 常被描述为一种基于原型的语言——每个对象拥有一个原型对象 当试图访问一个对象的属性时,它不仅仅在该对象上搜寻,还会搜寻该对象的原型,以及该对象的原型的原型,依次层层向上搜索,直到找到一个…

09.list 容器

9、list 容器 功能: 将数据进行链式存储 链表(list)是一种物理存储单元上非连续的存储结构,数据元素的逻辑顺序是通过链表中的指针链接实现的 链表的组成: 链表由一系列结点组成 结点的组成: 一个是存…

羊大师解答,小孩是喝羊奶好还是牛奶好

小孩是喝羊奶还是牛奶好,这是一个经常让父母头疼的问题。羊奶和牛奶都是优质的乳制品,含有丰富的蛋白质、钙和维生素等营养成分,对小孩的生长发育都有重要作用。然而,从营养角度来看,两者还是有一些差异的。 羊奶和牛…

YoloV8改进策略:AAAI 2024 最新的轴向注意力| 即插即用,改进首选|全网首发,包含数据集和代码,开箱即用!

摘要 本文提出了一种名为Multi-scale Cross-axis Attention(MCA)的方法,用于解决医学图像分割中的多尺度信息和长距离依赖性问题。该方法基于高效轴向注意力,通过计算两个平行轴向注意力之间的双向交叉注意力,更好地捕获全局信息。为了处理病变区域或器官的个体尺寸和形状…

tamarin manual总结笔记2(tamarin实例)

最初的例子 我们将从一个简单的协议示例开始&#xff0c;该协议仅由两条消息组成&#xff0c;在这里以所谓的Alice-and-Bob表示法编写: C -> S: aenc(k, pkS) C <- S: h(k)在该协议中&#xff0c;客户端C生成一个新的对称密钥k&#xff0c;用服务器S的公钥pkS (aenc代表…

k8s 中部署Jenkins

创建namespace apiVersion: v1 kind: Namespace metadata:name: jenkins创建pv以及pvc kind: PersistentVolume apiVersion: v1 metadata:name: jenkins-pv-volumenamespace: jenkinslabels:type: localapp: jenkins spec:#storageClassName: manualcapacity:storage: 5Giacc…

spring基于Xml管理bean---Ioc依赖注入:对象类型属性赋值(1)----外部bean的引入(bean和bean之间的引入)

文章目录 注入普通属性的方式1、set方法注入2、构造器&#xff08;构造方法&#xff09;注入 总结&#xff1a;注入对象类型属性 注入普通属性的方式 1、set方法注入 2、构造器&#xff08;构造方法&#xff09;注入 总结&#xff1a; set方法注入和构造器方法的注入&#…

用python实现adaboost算法例题

1.实验目的 1.会用Python提供的方法对数据进行预处理 2&#xff0e;会用python实现adaboost算法 2.设备与环境 Jupyter notebook 3.实验原理 4.实验内容 AdaBoost先初始化样本权值分布&#xff0c;并从初始训练集训练出一个基学习器&#xff0c;再根据这个基学习器的分类结…

AI时代Python量化交易实战:ChatGPT引领新时代

文章目录 《AI时代Python量化交易实战&#xff1a;ChatGPT让量化交易插上翅膀》关键点内容简介作者简介购买链接 《AI时代架构师修炼之道&#xff1a;ChatGPT让架构师插上翅膀》关键点内容简介作者简介 赠书活动 《AI时代Python量化交易实战&#xff1a;ChatGPT让量化交易插上翅…