OpenCV-Python(21):轮廓特征及周长、面积凸包检测和形状近似

2. 轮廓特征

        轮廓特征是指由轮廓形状和结构衍生出来的一些特征参数。这些特征参数可以用于图像识别、目标检测和形状分析等应用中。常见的轮廓特征包括:

  1. 面积:轮廓所包围的区域的面积。
  2. 周长:轮廓的周长,即轮廓线的长度。
  3. 弧长:轮廓线的弧长,即轮廓的长度。
  4. 轮廓矩:轮廓的几何矩,用于描述轮廓的形状。
  5. 轮廓重心:轮廓所包围区域的重心坐标。
  6. 外接矩形:能够完全包围轮廓的矩形。
  7. 最小外接矩形:能够紧密包围轮廓的矩形,且角度与轮廓的方向一致。
  8. 外接圆:能够完全包围轮廓的圆。
  9. 最小外接圆:能够紧密包围轮廓的圆。
  10. 椭圆拟合:能够最好地拟合轮廓的椭圆。
  11. 凸包:能够包围轮廓的最小凸多边形。
  12. 轮廓层级:描述轮廓的嵌套关系。

这些轮廓特征可以通过OpenCV库的cv2.contourArea()cv2.arcLength()cv2.moments()cv2.boundingRect()cv2.minAreaRect()cv2.minEnclosingCircle()cv2.fitEllipse()cv2.convexHull()等函数来计算和获取,下面主要介绍一些常用的特征。

2.1 目标

  •  查找轮廓的不同特征,例如面积、周长、重心、边界框等。

  •  学习和掌握轮廓相关函数

2.2 矩特征

        图像矩是一种描述图像几何特征的数学工具,用于描述图像的形状、位置和分布等信息,以帮助我们计算图像的质心、面积等。图像矩可以用于图像识别、目标检测、形状分析等应用中。图像矩的计算是基于图像像素的灰度值进行的,常见的图像矩包括原点矩、中心矩和归一化矩。使用OpenCV库的cv2.moments()函数可以计算图像的矩。该函数接受一个二值化的图像作为输入,并返回一个包含各种矩的字典。可以通过字典的键来获取不同的矩。以下是一个示例代码,演示如何计算图像的矩:

import cv2# 读取图像
image = cv2.imread("image.jpg")# 灰度化
gray = cv2.cvtColor(image, cv2.COLOR_BGR2GRAY)# 二值化
ret, thresh = cv2.threshold(gray, 127, 255, cv2.THRESH_BINARY)# 计算矩
moments = cv2.moments(thresh)# 获取原点矩
m00 = moments["m00"]
m10 = moments["m10"]
m01 = moments["m01"]# 获取中心矩
cx = int(m10 / m00)
cy = int(m01 / m00)# 获取归一化矩
nu20 = moments["nu20"]
nu02 = moments["nu02"]
nu11 = moments["nu11"]# 打印矩的值
print("m00:", m00)
print("m10:", m10)
print("m01:", m01)
print("cx:", cx)
print("cy:", cy)
print("nu20:", nu20)
print("nu02:", nu02)
print("nu11:", nu11)

上述代码中,首先读取了一张图像,并对其进行了灰度化和二值化处理。然后使用cv2.moments()函数计算了图像的矩。通过字典的键可以获取不同的矩的值。在示例代码中,获取了原点矩的值(m00、m10和m01)、中心矩的值(cx和cy)以及归一化矩的值(nu20、nu02和nu11)。最后打印了这些矩的值。需要注意的是,图像矩对图像的形状和位置非常敏感,因此在计算图像矩之前需要对图像进行预处理,如灰度化、二值化等。

注意:根据这些矩的值,我们可以计算出对象的重心,

cx = int(M['m10']/M['m00'])
cy = int(M['m01']/M['m00'])

2.3 轮廓面积

        轮廓面积是指闭合轮廓所包围的区域的面积,可以用来描述对象的大小。在OpenCV中,可以使用cv2.contourArea()函数计算轮廓的面积,也可以使用0阶矩(M['m00'])计算得到。

area = cv2.contourArea(cnt)

cv2.contourArea()函数接受一个轮廓作为输入,并返回轮廓的面积。该函数的语法如下:

area = cv2.contourArea(contour)

其中,contour是一个包含轮廓点的数组。可以通过cv2.findContours()函数找到图像中的轮廓,并提取其中的某个轮廓来计算面积。以下是一个示例代码,演示如何计算轮廓的面积:

import cv2# 读取图像
image = cv2.imread("image.jpg")# 灰度化
gray = cv2.cvtColor(image, cv2.COLOR_BGR2GRAY)# 二值化
ret, thresh = cv2.threshold(gray, 127, 255, cv2.THRESH_BINARY)# 查找轮廓
contours, hierarchy = cv2.findContours(thresh, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)# 遍历轮廓
for contour in contours:# 计算轮廓面积area = cv2.contourArea(contour)# 打印轮廓面积print("Contour area:", area)

在上述代码中,首先读取了一张图像,并对其进行了灰度化和二值化处理。然后使用cv2.findContours()函数找到图像中的轮廓。返回的contours是一个包含所有轮廓的列表。接下来,遍历轮廓列表,并使用cv2.contourArea()函数计算每个轮廓的面积。最后打印了每个轮廓的面积。需要注意的是,cv2.contourArea()函数计算的是轮廓的面积,而不是对象的实际面积。因此,在使用该函数之前,需要对图像进行预处理,如灰度化、二值化等,以确保轮廓能够正确识别。

2.4 轮廓周长

        轮廓周长是指闭合轮廓的长度,可以用来描述对象的形状。在OpenCV中,可以使用cv2.arcLength()函数计算轮廓的周长。cv2.arcLength()函数接受一个轮廓作为输入,并返回轮廓的周长。该函数的语法如下:

perimeter = cv2.arcLength(contour, closed)

其中,contour是一个包含轮廓点的数组,closed是一个布尔值,指示轮廓是否是闭合的。如果轮廓是闭合的,则closedTrue;如果轮廓是开放的,则closedFalse。以下是一个示例代码,演示如何计算轮廓的周长:

import cv2# 读取图像
image = cv2.imread("image.jpg")# 灰度化
gray = cv2.cvtColor(image, cv2.COLOR_BGR2GRAY)# 二值化
ret, thresh = cv2.threshold(gray, 127, 255, cv2.THRESH_BINARY)# 查找轮廓
contours, hierarchy = cv2.findContours(thresh, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)# 遍历轮廓
for contour in contours:# 计算轮廓周长perimeter = cv2.arcLength(contour, True)# 打印轮廓周长print("Contour perimeter:", perimeter)

在上述代码中,首先读取了一张图像,并对其进行了灰度化和二值化处理。然后使用cv2.findContours()函数找到图像中的轮廓。返回的contours是一个包含所有轮廓的列表。接下来,遍历轮廓列表,并使用cv2.arcLength()函数计算每个轮廓的周长。最后打印了每个轮廓的周长。需要注意的是,cv2.arcLength()函数计算的是轮廓的周长,而不是对象的周长。因此,在使用该函数之前,需要对图像进行预处理,如灰度化、二值化等,以确保轮廓能够正确识别。

2.5 轮廓近似

        在OpenCV中,可以使用cv2.approxPolyDP()函数对轮廓进行近似处理。轮廓近似可以将复杂的轮廓形状简化为更简单的几何形状,如直线或曲线。cv2.approxPolyDP()函数接受一个轮廓作为输入,并返回一个近似的轮廓。该函数的语法如下:

approx = cv2.approxPolyDP(curve, epsilon, closed)

其中,curve是一个包含轮廓点的数组,epsilon是指定近似精度的参数,closed是一个布尔值,指示轮廓是否是闭合的。如果轮廓是闭合的,则closedTrue;如果轮廓是开放的,则closedFalse。将轮廓形状近似到另外一种由更少点组成的轮廓形状,新轮廓的点的数目由我们设定的准确度来决定。实现这一功能主要使用的是Douglas-Peucker算法(你可以到维基百科获得更多此算法的细节)。为了帮助理解,假设我们需要在一幅图像中查找一个矩形,但是由于图像的种种原因,我们不能得到一个完美的矩形,而是一个(坏形状),如下图所示:

现在就可以使用这个函数来近似这个形状了。这个函数的第二个参数叫epsilon,它是从原始轮廓到近似轮廓的最大距离。它是一个准确度参数。选择一个好的epsilon 对于得到满意结果非常重要。

epsilon = 0.1*cv2.arcLength(cnt,True)
approx = cv2.approxPolyDP(cnt,epsilon,True)

下面第二幅图中的绿线是当epsilon = 10% 时得到近似轮廓,第三幅图是当epsilon = 1% 时得到的近似轮廓。第三个参数设定弧线是否闭合。

以下是一个示例代码,演示如何对轮廓进行近似处理:

import cv2# 读取图像
image = cv2.imread("image.jpg")# 灰度化
gray = cv2.cvtColor(image, cv2.COLOR_BGR2GRAY)# 二值化
ret, thresh = cv2.threshold(gray, 127, 255, cv2.THRESH_BINARY)# 查找轮廓
contours, hierarchy = cv2.findContours(thresh, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)# 遍历轮廓
for contour in contours:# 近似轮廓epsilon = 0.01 * cv2.arcLength(contour, True)approx = cv2.approxPolyDP(contour, epsilon, True)# 绘制近似轮廓cv2.drawContours(image, [approx], 0, (0, 255, 0), 2)

在上述代码中,首先读取了一张图像,并对其进行了灰度化和二值化处理。然后使用cv2.findContours()函数找到图像中的轮廓。返回的contours是一个包含所有轮廓的列表。接下来,遍历轮廓列表,并使用cv2.arcLength()函数计算每个轮廓的周长。然后根据周长计算一个近似精度epsilon,并使用cv2.approxPolyDP()函数对轮廓近似处理。最后使用cv2.drawContours()函数绘制近似轮廓。需要注意的是,轮廓近似处理是一种对轮廓进行简化的方法,可以减少轮廓点的数量,从而降低计算的复杂性。但近似精度epsilon的选择需要根据具体情况进行调整,过大的epsilon会导致近似结果不准确,而过小的epsilon会导致近似结果过于复杂。因此,在使用cv2.approxPolyDP()函数进行轮廓近似处理时,需要根据实际情况选择合适的epsilon值。

2.6 凸包

        在计算机视觉中,凸包(Convex Hull)是指一个包围一组点的最小凸多边形。凸多边形是一个所有内角均小于180度的多边形。凸包与轮廓近似相似,但不同,然有些情况下它们给出的结果是一样的。OpenCV中提供了cv2.convexHull()函数来计算给定点集的凸包,同时检测一个曲线是否具有凸性缺陷并能纠正缺陷。一般来说,凸性曲线总是凸出来的或者至少是平的。如果有地方凹下去了,就叫做凸性缺陷。例如下图中的手。红色曲线显示了手的凸包,凸性缺陷被双箭头标出来了。


关该cv2.convexHull()函数的语法如下:

hull = cv2.convexHull(points, clockwise, returnPoints)

其中,points是一个包含点集的数组,clockwise是一个布尔值,用于指定凸包的顺序,如果为True,则返回的凸包按顺时针方向排序;如果为False,则返回的凸包按逆时针方向排序;returnPoints是一个布尔值,用于指定返回的凸包是点的坐标还是索引,默认为True,表示返回点的坐标。

要获得上图的凸包,下面的命令就够了:

hull = cv2.convexHull(cnt)

但是如果你想获得凸性缺陷,需要把returnPoints 设置为False。以上面的矩形为例,首先我们找到他的轮廓cnt。现在我把returnPoints 设置为True 查找凸包,我得到下列值:

[[[234 202]], [[ 51 202]], [[ 51 79]], [[234 79]]],其实就是矩形的四个角点。现在把returnPoints 设置为False,我得到的结果是[[129],[ 67],[ 0],[142]],他们是轮廓点的索引。例如cnt[129] = [[234, 202]]这与前面我们得到结果的第一个值是一样的。

以下是一个示例代码,演示如何使用cv2.convexHull()函数计算凸包:

import cv2
import numpy as np# 创建一组点
points = np.array([[10, 10], [10, 100], [100, 10], [100, 100], [50, 50]])# 计算凸包
hull = cv2.convexHull(points)# 绘制凸包
image = np.zeros((200, 200, 3), dtype=np.uint8)
cv2.polylines(image, [hull], True, (0, 255, 0), 2)# 显示图像
cv2.imshow("Convex Hull", image)
cv2.waitKey(0)
cv2.destroyAllWindows()

在上述代码中,首先创建了一个包含5个点的数组points。然后使用cv2.convexHull()函数计算凸包,并将结果保存在hull中。接下来,创建一个空白图像image,并使用cv2.polylines()函数绘制凸包。最后,显示图像。

需要注意的是,cv2.convexHull()函数返回的凸包是一个包含点的数组,可以通过cv2.polylines()函数绘制凸包。另外,可以使用cv2.isContourConvex()函数检查一个轮廓是否是凸的。如果返回True,则表示轮廓是凸的;如果返回False,则表示轮廓是非凸的。凸包在计算机视觉中有广泛的应用,如图像分割、形状匹配等。凸包可以帮助我们简化复杂的形状,并提取出形状的关键特征。

2.7 凸性检测

        函数cv2.isContourConvex() 可以可以用来检测一个曲线是不是凸的。它只能返回True 或False。没什么大不了的。

k = cv2.isContourConvex(cnt)

2.8 边界矩形

        通常有两类边界矩形:直边界矩形、旋转边界矩形

直边界矩形 

        一个直矩形(就是没有旋转的矩形)。它不会考虑对象是否旋转。所以直边界矩形的面积不是最小的。可以使用函数cv2.boundingRect() 查找得到。(x,y)为矩形左上角的坐标,(w,h)是矩形的宽和高。

x,y,w,h = cv2.boundingRect(cnt)
img = cv2.rectangle(img,(x,y),(x+w,y+h),(0,255,0),2)
旋转边界矩形

        这个边界矩形是积最小的,因为它考虑了对象的旋转。用到的函数为cv2.minAreaRect()。返回的是一个Box2D 结构,其中包含矩形左上角角点的坐标(x,y),矩形的宽和高(w,h)以及旋转角度。但是要绘制这个矩形需要矩形的4 个顶点,可以通过函数cv2.boxPoints() 获得。

x,y,w,h = cv2.boundingRect(cnt)
img = cv2.rectangle(img,(x,y),(x+w,y+h),(0,255,0),2)

把这两中边界矩形显示在下图中,其中绿色的为直矩形,红的为旋转矩形。

2.9 最小外接圆

        函数cv2.minEnclosingCircle() 可以帮我们找到一个对象的外接圆。它是所有能够包括对象的圆中面积最小的一个。

(x,y),radius = cv2.minEnclosingCircle(cnt)
center = (int(x),int(y))
radius = int(radius)
img = cv2.circle(img,center,radius,(0,255,0),2)

2.10 椭圆拟合

        使用的函数为cv2.ellipse(),返回值其实就是旋转边界矩形的内切圆。

ellipse = cv2.fitEllipse(cnt)
im = cv2.ellipse(im,ellipse,(0,255,0),2)

2.11 直线拟合

        我们可以根据一组点拟合出一条直线,同样我们也可以为图像中的白色点拟合出一条直线。

rows,cols = img.shape[:2]
[vx,vy,x,y] = cv2.fitLine(cnt, cv2.DIST_L2,0,0.01,0.01)
lefty = int((-x*vy/vx) + y)
righty = int(((cols-x)*vy/vx)+y)
img = cv2.line(img,(cols-1,righty),(0,lefty),(0,255,0),2)

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

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

Linux 线程概念

文章目录 前言线程的概念线程的操作操作的原理补充与说明 前言 ① 函数的具体说明被放在补充与说明部分 ② 只说些基础概念和函数使用 线程的概念 网络回答:Linux 线程是指在 Linux 操作系统中创建和管理的轻量级执行单元。线程是进程的一部分,与进程…

Web漏洞—安全评估基础知识

一个安全评估的过程,可以简单地分为4个阶段:资产等级划分、威胁分析、风险分析、确认解决方案。 一般来说,按照这个过程来实施安全评估,在结果上不会出现较大的问题。这个实施的过程是层层递进的,前后之间有因果关系。 资产等级划分 资产等级…

【CSS3】第4章 CSS3选择器

学习目标 熟悉属性选择器的用法,了解不同属性选择器的功能。 掌握关系选择器的用法,能够使用关系选择器选取父标签中嵌套的子标签。 掌握结构化伪类选择器的用法,能够使用不同功能的结构化伪类选择器精准控制标签样式。 掌握状态化伪类选择…

HCIA-Datacom题库(自己整理分类的)——OSPF协议多选

ospf的hello报文功能是 邻居发现 同步路由器的LSDB 更新LSA信息 维持邻居关系 下列关于OSPF区域描述正确的是 在配置OSPF区域正确必须给路由器的loopback接配置IP地址 所有的网络都应在区域0中宣告 骨干区域的编号不能为2 区域的编号范围是从0.0.0.0到255.255.255.255…

边缘计算网关:在智慧储能系统中做好储能通信管家

背景 目前储能系统主要由储能单元和监控与调度管理单元组成,储能单元包含储能电池组(BA)、电池管理系统(BMS)、储能变流器(PCS)等;监控与调度管理单元包括中央控制系统(MGCC)、能量管理系统(EMS)等。 2021年8月,国家发改委发布《电化学储能…

新版ONENET的物联网环境调节系统(esp32+onenet+微信小程序)

新版ONENET的物联网环境调节系统(esp32onenet微信小程序) 好久没用onenet突然发现它大更新了,现在都是使用新版的物联网开放平台,只有老用户还有老版的多协议接入,新用户是没有的,所以我顺便更新一下新的开…

大厂前端面试题总结(百度、字节跳动、腾讯、小米.....),附上热乎面试经验!

先简单介绍下自己,我“平平无奇小天才”一枚,毕业于南方普通985普通学生,有幸去百度、字节面试,感觉大公司就是不一样,印象最深的是字节,所以有必要总结一下面试经验,以及面试中遇到的一些问题&…

html-css-js使用axios和ajax获取接口并携带请求头+获取输入框或选择器内容

需求:使用axios或者Ajax获取接口,有些需要获取到输入框,或者选择器内容之后传给接口,也就是写了几种不同请求的方法,网上有很多方法,本文章算是个归纳吧。 一、axios请求传参请求头 1.github下载axios 我…

智安网络|实现安全与网络功能一体化:SASE的全新安全策略

随着企业信息化和数字化程度的不断提升,网络安全面临着前所未有的挑战。传统的网络安全模式已经无法满足日益复杂的安全需求。在这一背景下,安全访问服务边缘(SASE)崭露头角,并逐渐成为新一代网络安全架构的关键概念。…

嵌入式开发——ADC开发

学习目标 了解ADC开发流程掌握采样方式能够使用ADC进行芯片内部通道进行采样能够使用ADC对外部电路进行采样学习内容 GD32F4的ADC 特点: 16个外部模拟输入通道;1个内部温度传感通道(VSENSE);1个内部参考电压输入通道(VREFINT);1个外部监测电池VBAT供电引脚输入通道。ADC开…

自动化测试与功能测试

什么是自动化测试? 自动化测试是指利用软件测试工具自动实现全部或部分测试,它是软件测试的一个重要组成 部分,能完成许多手工测试无法实现或难以实现的测试。能够正确、合理地实施自动测试,可以 快速、全面地对软件进行测试,从…

JVM篇:JVM内存结构

程序计数器 程序计数器英文名叫:Program Counter Register 作用:用来记录下一条jvm指令的地址行号。 先来查看一段jvm指令,这些指令对应的java代码就是输出1-5 操作系统运行该Java程序时具体流程如下 语言解释:源文件通过编译转…

水果软件2024FL Studio21.3mac苹果中文版

FL STUDIO21发布,提供您一直在等待的出色工作流程功能。通过新效果、多个播放列表曲目选择和无所畏惧的撤消一切编辑,将您的音乐带入2024年。FL Studio21中文完整版是一个功能齐全、开放式架构的PC音乐创作和制作环境。它具有基于音乐音序器的图形用户界…

OFDM——PAPR减小

文章目录 前言一、PAPR 减小二、MATLAB 仿真1、OFDM 信号的 CCDF①、MATLAB 源码②、仿真结果 2、单载波基带/通频带信号的 PAPR①、MATLAB 源码②、仿真结果 3、时域 OFDM 信号和幅度分布①、MATLAB 源码②、仿真结果 4、Chu 序列和 IEEE802.16e 前导的 PAPR①、MATLAB 源码②…

手撕测试tcp服务器效率工具——以epoll和io_uring对比为例

服务器性能测试介绍 服务器的性能测试主要包括2部分: 并发量。能容纳多大的连接效率。在不崩坏的情况下能对报文的处理效率。 本文主要进行效率测试,看看基于epoll模型和io_uring模型的tcp服务器,谁的效率更高。 测试思路 客户端&#x…

字节跳动 MapReduce - Spark 平滑迁移实践

摘要:本文整理自字节跳动基础架构工程师魏中佳在本次 CommunityOverCode Asia 2023 中的《字节跳动 MapReduce - Spark 平滑迁移实践》主题演讲。 随着字节业务的发展,公司内部每天线上约运行 100万 Spark 作业,与之相对比的是,线…

mac 生成 本地.ssh

输入下面命令行 ssh-keygen 默认回车得到下面的 Generating public/private rsa key pair. Enter file in which to save the key (/Users/{用户名}/.ssh/id_rsa): Enter passphrase (empty for no passphrase): Enter same passphrase again: Your identification has be…

中央集成式架构量产时代,openVOC方案将引发软件开发模式变革

2024年,中央计算区域控制架构正式进入规模化量产周期,汽车智能化正式迈入2.0时代,产业生态、应用创新、开发模式都将迎来巨大变革。 同时,随着ChatGPT引发的AIGC领域的爆发式增长,人工智能技术掀起全球万亿级信息化应…

MySQL进阶之(一)逻辑架构

一、逻辑架构 1.1 逻辑架构剖析1.1.1 连接层1.1.2 服务层01、基础服务组件02、SQL Interface:SQL 接口03、Parser:解析器04、Optimizer:查询优化器05、Caches & Buffers: 查询缓存组件 1.1.3 引擎层1.1.4 存储层1.1.5 总结 1.…

Android画布Canvas矩阵Matrix放大裁剪Rect区域的Bitmap,Kotlin

Android画布Canvas矩阵Matrix放大裁剪Rect区域的Bitmap,Kotlin private fun mydraw() {val originBmp BitmapFactory.decodeResource(resources, R.mipmap.pic).copy(Bitmap.Config.ARGB_8888, true)val newBmp Bitmap.createBitmap(originBmp.width, originBmp.h…