OpenCV 图像轮廓检测

news2024/10/1 5:35:04

        本文是OpenCV图像视觉入门之路的第15篇文章,本文详细的介绍了图像轮廓检测的各种操作,例如:轮廓检索模式、轮廓逼近算子等操作。

        图像轮廓是具有相同颜色或灰度的连续点的曲线,轮廓在形状分析和物体的检测和识别中很有用。图像轮廓的作用:图形分析、物体的检测和识别。

        需要注意的是:为了检测的准确性,需要先对图像进行二值化或Canny操作; 画轮廓时会修改输入的图像,如果之后想继续使用原始图像,应该将原始图像存储到其他变量中。

OpenCV 图像轮廓检测目录

1 查找轮廓

2 绘制轮廓

3 轮廓的面积和周长

4 多边形逼近

5 凸包

6 外接矩形


1 查找轮廓

cv2.findContours(image, mode, method, contours, hierarchy, offset)

  • image 原图像
  • mode 查找轮廓的模式
  •         RETR_EXTERNAL= 0,表示只检测外围轮廓 。
  •         RETR_LIST = 1,检测的轮廓不建立等级关系,即检测所有轮廓。
  •         RETP_CCOMP = 2,每层最多两级,从小到大,从里到外。
  •         RETP_TREE = 3,按照树型存储轮廓,从小到大,从右到左。
  • method 轮廓近似方法,也叫ApproximationMode
  •         CHAIN_APPROX_NONE:保存所有轮廓上的点
  •         CHAIN_APPROX_SIMPLE:只保存角点。比如四边形,只保留四边形的四个角,存储信息少,比较常用
  • 返回 contours 和 hierarchy ,即轮廓和层级。

import cv2
import numpy as np
from numpy import unicode

d

if __name__ == '__main__':
    img = cv2.imread('D:/Jupyter_Notebooks/1.2.png') #, cv2.IMREAD_GRAYSCALE

    # 转变成单通道
    gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)
    # 二值化,第一个返回值是执行的结果和状态是否成功,第二个返回值才是真正的图片结果
    ret, binary = cv2.threshold(gray, 150, 255, cv2.THRESH_BINARY)
    # 轮廓查找,第一个返回值是轮廓,第二个是层级
    contours, hierarchy = cv2.findContours(binary, cv2.RETR_TREE, cv2.CHAIN_APPROX_SIMPLE)
    # 绘制轮廓
    cv2.drawContours(img, contours, -1, (0, 0, 255), 1)  # 改变的是img这张图


    cv2.imshow('img', img)
    cv2.imshow('binary', binary)

    cv2.waitKey(0)
    cv2.destroyAllWindows()

2 绘制轮廓

  • cv2.drawContours(image, contours, contourIdx, color, thickness, lineType, hierarchy, maxLevel, offset)
  • image:要绘制的轮廓图像
  • contours:轮廓点
  • contourIdx:要绘制的轮廓的编号,-1 表示绘制所有轮廓
  • color:轮廓的颜色,如 (0, 0, 255) 表示红色
  • thickness:线宽,-1表示全部填充
import cv2
import numpy as np
from numpy import unicode



if __name__ == '__main__':
    img = cv2.imread('D:/Jupyter_Notebooks/1.2.png') #, cv2.IMREAD_GRAYSCALE

    img_copy = img.copy()
    gray = cv2.cvtColor(img_copy, cv2.COLOR_BGR2GRAY)

    # 阈值 结果
    thresh, binary = cv2.threshold(gray, 150, 255, cv2.THRESH_BINARY)
    # 轮廓 层级
    contours, hierarchy = cv2.findContours(binary, cv2.RETR_TREE, cv2.CHAIN_APPROX_SIMPLE)
    # 绘制轮廓
    cv2.drawContours(img_copy, contours, -1, (0, 0, 255), 2)

    cv2.imshow('img', np.hstack((img, img_copy)))

    cv2.waitKey(0)
    cv2.destroyAllWindows()

3 轮廓的面积和周长

        轮廓面积是指每个轮廓中所有的像素点围成区域的面积,单位为像素。轮廓面积是轮廓重要的统计特性之一,通过轮廓面积的大小可以进一步分析每个轮廓隐含的信息,例如通过轮廓面积区分物体大小,进而可以识别不同的物体。在查找轮廓后,可能会有很多细小的轮廓,我们可以通过轮廓的面积进行过滤。

  • 面积        contourArea()        cv2.contourArea(contour, oriented)               contour:轮廓
  • 周长         arcLength()        cv2.arcLength(curve, closed)                        curve:轮廓        closed:是否为闭合的轮廓
import cv2
import numpy as np
from numpy import unicode



if __name__ == '__main__':
    img = cv2.imread('D:/Jupyter_Notebooks/1.2.png') #, cv2.IMREAD_GRAYSCALE

    img_copy = img.copy()
    gray = cv2.cvtColor(img_copy, cv2.COLOR_BGR2GRAY)

    # 阈值 结果
    thresh, binary = cv2.threshold(gray, 150, 255, cv2.THRESH_BINARY)
    # 轮廓 层级
    contours, hierarchy = cv2.findContours(binary, cv2.RETR_TREE, cv2.CHAIN_APPROX_SIMPLE)
    # 绘制轮廓
    cv2.drawContours(img_copy, contours, -1, (0, 0, 255), 2)
    # 轮廓面积
    area = cv2.contourArea(contours[1])
    print('area:', area)
    # 轮廓周长
    perimeter = cv2.arcLength(contours[1], closed=True)
    print('perimeter:', perimeter)

    cv2.imshow('img', np.hstack((img, img_copy)))

    cv2.waitKey(0)
    cv2.destroyAllWindows()

4 多边形逼近

        findContours 后的轮廓信息 contours 可能过于复杂不平滑,可以用 approxPolyDP 函数对该多边形曲线做适当近似,这就是轮廓的多边形逼近。approxPolyDP 就是以多边形去逼近轮廓,采用的是 Douglas-Peucker(方法名中的DP)。 DP算法原理比较简单,核心就是不断找多边形最远的点加入,形成新的多边形,直到最短距离小于指定的精度。

  • approxPolyDP()用法:        cv2.approxPolyDP(curve, epsilon, closed, approxCurve)
  • curve:要近似逼近的轮廓
  • epsilon:即DP算法使用的阈值
  • closed:轮廓是否闭合
import cv2
import numpy as np
from numpy import unicode



if __name__ == '__main__':
    img = cv2.imread('D:/Jupyter_Notebooks/1.3.png') #, cv2.IMREAD_GRAYSCALE

    img_ori = img.copy()
    gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)

    # 二值化:阈值 结果
    ret, binary = cv2.threshold(gray, 150, 255, cv2.THRESH_BINARY)
    # 查找轮廓 轮廓 层级
    contours, hierarchy = cv2.findContours(binary, 0, cv2.CHAIN_APPROX_SIMPLE)
    # 绘制轮廓
    cv2.drawContours(img, contours, 1, (0, 0, 255), 2)
    # 多边形逼近
    approx = cv2.approxPolyDP(contours[1], 10, True)
    # 把多边形逼近的轮廓画出来
    cv2.drawContours(img, [approx], -1, (200, 255, 0), 2)

    cv2.imshow('hand', np.hstack((img_ori, img)))

    cv2.waitKey(0)
    cv2.destroyAllWindows()

RETR_EXTERNAL=0表示值检测外围轮廓

 RETR_LIST=1检测的轮廓不建立等级关系,检测所有的轮廓

5 凸包

        多边形逼近是轮廓的高度近似,但有些时候,我们希望使用一个多边形的凸包来简化它。凸包和多边形逼近很类似,只不过它是物体最外层的凸多边形。凸包指的是完全包含原有轮廓,并且仅由轮廓上的点所构成的多边形。凸包上的每一处都是凸的,即在凸包内连接任意两点的直线都在凸包的内部。在凸包内,任意连续三个点的内角小于180°。

  • convexHull()用法:cv2.convexHull(points, hull, clockwise, returnPoints)
  • points:轮廓
  • clockwise:顺时针绘制
import cv2
import numpy as np
from numpy import unicode



if __name__ == '__main__':
    img = cv2.imread('D:/Jupyter_Notebooks/1.3.png') #, cv2.IMREAD_GRAYSCALE

    img_ori = img.copy()
    gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)

    # 二值化:阈值 结果
    ret, binary = cv2.threshold(gray, 150, 255, cv2.THRESH_BINARY)
    # 查找轮廓 轮廓 层级
    contours, hierarchy = cv2.findContours(binary, 0, cv2.CHAIN_APPROX_SIMPLE)
    # 绘制轮廓
    cv2.drawContours(img, contours, 1, (0, 0, 255), 2)
    # 多边形逼近
    approx = cv2.approxPolyDP(contours[1], 10, True)
    # 把多边形逼近的轮廓画出来
    cv2.drawContours(img, [approx], -1, (0, 255, 0), 2)
    # 计算凸包
    hull = cv2.convexHull(contours[1])
    cv2.drawContours(img, [hull], -1, (255, 0, 0), 2)

    cv2.imshow('hand', np.hstack((img_ori, img)))

    cv2.waitKey(0)
    cv2.destroyAllWindows()

6 外接矩形

        外接矩形分为最小外接矩形和最大外接矩形。

下图中,红色部分为最小外接矩形,绿色部分为最大外接矩形:

minAreaRect()用法: 最小外接矩阵                cv2.minAreaRect(points)

  • points:轮廓
  • 返回元组,内容是一个旋转矩形(RotatedRect)的参数:矩形的起始坐标x,y,矩形的宽度和高度,矩形的旋转角度

boundingRect()用法:最大外接矩阵        cv2.boundingRect(points)

  • points:轮廓
import cv2
import numpy as np
from numpy import unicode



if __name__ == '__main__':
    img = cv2.imread('D:/Jupyter_Notebooks/1.3.png') #, cv2.IMREAD_GRAYSCALE

    img_ori = img.copy()
    gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)

    # 二值化:阈值 结果
    ret, binary = cv2.threshold(gray, 150, 255, cv2.THRESH_BINARY)
    # 查找轮廓 轮廓 层级
    contours, hierarchy, = cv2.findContours(binary, 0, cv2.CHAIN_APPROX_SIMPLE)

    # 最小外接矩形
    rect = cv2.minAreaRect(contours[1])
    box = cv2.boxPoints(rect)
    # 四舍五入
    box = np.round(box).astype('int64')
    # 绘制最小外接矩形
    cv2.drawContours(img, [box], 0, (255, 0, 0), 2)

    # 最大外接矩形
    x, y, w, h = cv2.boundingRect(contours[1])
    cv2.rectangle(img, (x, y), (x + w, y + h), (0, 0, 255), 2)

    cv2.imshow('hand', np.hstack((img_ori, img)))

    cv2.waitKey(0)
    cv2.destroyAllWindows()

 

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

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

相关文章

预训练BERT

与PTB数据集相比,WikiText-2数据集保留了原来的标点符号、大小写和数字,并且比PTB数据集大了两倍多。 我们可以任意访问从WikiText-2语料库中的一对句子生成的预训练(遮蔽语言模型和下一句预测)样本。 原始的BERT有两个版本&…

《Python机器学习》基础代码

1,要学习Python机器学习,第一步就是读入数据,这里我们以读入excel的数据为例,利用jupyter notebook来编码,具体教程看这个视频 推荐先上传到jupyter notebook,再用名字.xlsx来导入 Jupyter notebook导入Excel数据的两种方法介绍_哔哩哔哩_bilibili 2,…

MyBatis - 06 - MyBatis获取参数值的两种方式(重要)

文章目录1、回顾JDBC原生的获取参数值的方式2、MyBatis获取参数值的两种方式3、MyBatis获取参数值的五种情况情况1: Mapper接口方法的参数为单个字面量类型的参数ParameterMapper接口代码测试类代码ParameterMapper.xml配置方式1:${}ParameterMapper.xml配置方式2:#…

[8]云计算概念、技术与架构Thomas Erl-第九章 云管理机制|week9|10月29日

9.0引言 基于云的IT资源需要被建立、配置、维护和监控。 远程管理系统 资源管理系统 SLA管理系统 计费管理系统 这些系统通常会提供整合的API,并能够以个别产品、定制应用或者各种组合产品套装和多功能应用的形式提供给用户。 9.1 远程管理系统 远程管理系统…

「回顾RKDC2023」飞凌嵌入式受邀亮相第七届瑞芯微开发者大会

2023年2月23-24日,第七届瑞芯微开发者大会(RKDC2023)在福州隆重举行,飞凌嵌入式作为瑞芯微生态合作伙伴受邀参会,并与数千名开发者科技公司代表及行业领袖共同聚焦行业新兴产品需求,探讨新硬件发展趋势&…

webpack实战,手写loader和plugin

序言 对于 webpack 来说, loader 和 plugin 可以算是需求程度最为广泛的配置项了。但是呢,单单止步于配置可能还不够。如果我们自己有时候想要 diy 一个需求,但是 webpack 又没有相关的 loader 和 plugin 。那这个时候我们可能就得开始造点轮…

【JAVA程序设计】(C00105)基于SSM大学在校生职业走向调查分析系统-有文档

基于SSM大学在校生职业走向调查分析系统-有文档项目简介项目获取开发环境项目技术运行截图项目简介 基于ssm框架大学生在校生职业走向调查分析系统分为二个角色:系统管理员、用户 管理员角色包含以下功能: 调查课题管理、留言信息管理、在校学生管理、社…

华为OD机试题,用 Java 解【最小步骤数】问题

最近更新的博客 华为OD机试题,用 Java 解【停车场车辆统计】问题华为OD机试题,用 Java 解【字符串变换最小字符串】问题华为OD机试题,用 Java 解【计算最大乘积】问题华为OD机试题,用 Java 解【DNA 序列】问题华为OD机试 - 组成最大数(Java) | 机试题算法思路 【2023】使…

JavaSE-3 Java运行原理

一、Java的运行过程 🍎Java程序运行时,必须经过编译和运行两个步骤。 首先将后缀名为.java的源文件进行编译,最终生成后缀名为.class的字节码文件。然后Java虚拟机将字节码文件进行解释执行,并将结果显示出来。具体过程如下图所示。 🍉Java程序的运行过…

Java集合学习:LinkedList源码详解

前言 LinkedList在我们平时工作中使用频率非车高,底层是基于双向链表数据结构实现,下面从经常使用的几个方法来了解其原理。 正文 结构 我们先看下LinkedList的重要属性 /**存储链表数量*/transient int size 0;/**存储链表的头节点*/transient Node…

MTK平台使用Omnipeek分析空口协议讲解

讲解这个之前,我们先来了解下beacon/robe Request/Probe Response 三种帧 beacon帧 信标帧,由AP以一定的时间间隔周期性发出,以此来告诉外界自己无线网络的存在。 Beacon帧作为802.11中一个周期性的帧,Beacon周期调高,对应睡眠周期拉长,故节能(即越来休息100ms再起来…

Laravel框架04:视图与CSRF攻击

Laravel框架04:视图与CSRF攻击一、视图概述二、变量分配与展示三、模板中直接使用函数四、循环与分支语法标签五、模板继承、包含1. 继承2. 包含六、外部静态文件引入七、CSRF攻击概述八、从CSRF验证中排除例外路由一、视图概述 视图存放在 resources/views 目录下…

Verilog 学习第五节(串口接收部分)

小梅哥串口部分学习part2 串口通信接收原理串口通信接收程序设计与调试巧用位操作优化串口接收逻辑设计串口接收模块的项目应用案例串口通信接收原理 在采样的时候没有必要一直判断一个clk内全部都是高/低电平,如果采用直接对中间点进行判断的话,很有可能…

CISP注册信息安全专业人员证书

一、什么是“CISP”? 注册信息安全专业人员(Certified Information Security Professional,简称“CISP”),是安全行业最为权威的安全资格认证,由中国信息安全测评中心统一授权组织,中国信息安全测评中心授权培训机构进…

学习笔记之Vue中的Ajax(四)

Vue中的Ajax(四)Vue中的ajax一、解决开发环境Ajax跨越问题二、github 用户搜索案例2.1 准备工作2.2 静态页面2.3 实现动态组件2.4 注意细节三、vue 项目中常用的 2 个 Ajax 库3.1 axios3.2 vue-resource四、slot插槽(四)Vue中的aj…

计算结构体大小

计算结构体大小 目录计算结构体大小一. 结构体内存对齐1. 简介2. 嵌套结构体二. offsetof三. 内存对齐的意义四. 修改默认对齐数一. 结构体内存对齐 以字节(bety)为单位 1. 简介 对于结构体成员在内存里的存储,存在结构体的对齐规则&#…

代码随想录算法训练营day44 | 动态规划之完全背包 518. 零钱兑换 II 377. 组合总和 Ⅳ

day44完全背包基础知识问题描述举个栗子518. 零钱兑换 II1.确定dp数组以及下标的含义2.确定递推公式3.dp数组如何初始化4.确定遍历顺序5.举例推导dp数组377. 组合总和 Ⅳ1.确定dp数组以及下标的含义2.确定递推公式3.dp数组如何初始化4.确定遍历顺序5.举例来推导dp数组完全背包基…

安全合规之CVE-2016-2183

文章目录概述分析解决补充信息概述 安全部门脆弱性扫描到如下的风险漏洞要求系统上线必须要修复完毕。 不过我仔细的看了安全部门返回的报告,它是针对Windows Server 2019远程桌面端口进行风险报告…这是刷存在感了吗?哎,没有办法先做调查确…

高压放大器在声波谐振电小天线收发测试系统中的应用

实验名称:高压放大器在声波谐振电小天线收发测试系统中的应用研究方向:信号传输测试目的:声波谐振电小天线颠覆了传统电小天线以电磁波谐振作为理论基础的天线发射和接收模式,它借助声波谐振实现电磁信号的辐射或接收。因为同频的…

Spring Batch 综合案例实战-项目准备

目录 案例需求 分析 项目准备 步骤1:新开spring-batch-example 步骤2:导入依赖 步骤3:配置文件 步骤4:建立employee表与employe_temp表 步骤5:建立基本代码体系-domain-mapper-service-controller-mapper.xml …