机器视觉学习-day14-绘制图像轮廓

1. 轮廓的概念

轮廓是目标物体或者区域在图像外部的边界线通常由一系列像素点相连组成,这些像素点共同构成了一个封闭的形状,这样形状就是轮廓。

轮廓与边缘不同:

        轮廓是连续的,边缘可以连续也可以离散

        轮廓是完整的,边缘可以是完成的也可以不完整

轮廓主要分析物体的形态,比如计算物体的周长与面积;边缘作为图像特征使用,比如区分脸和手。

代码运行步骤:

图片输入→灰度化→二值化→寻找轮廓→绘制轮廓→图片输出

1.1 寻找轮廓

    # 4. 寻找轮廓# 返回值1:所有轮廓的点坐标,是一个list列表# 返回值2:轮廓的层级关系contours, hierarchy = cv2.findContours(image = image_np_thresh,  # 二值化之后的图像mode = cv2.RETR_EXTERNAL,  # 默认的轮廓查找方式method = cv2.CHAIN_APPROX_SIMPLE  # 默认的轮廓近似办法)

●contours 轮廓s
tuple类型,所有轮廓的点(每个元素是Numpy数组),可以通过[n-1]取出第n个轮廓的数据
●hierarchy 轮廓关系
如果没有检测到轮廓,则返回的数组为空,通常不会出现。
对于第n个轮廓:
○hierarchy[n][0]表示当前轮廓的后一条轮廓
○hierarchy[n][1]表示当前轮廓的前一条轮廓
○hierarchy[n][2]表示当前轮廓的第一个子轮廓
○hierarchy[n][3]表示当前轮廓的父轮廓
如果出现-1表示没有。
此参数较少使用,通常用于处理复杂画面效果。
●image 输入图像
要求8位的二值化图像,前景色白色,背景色黑色
●mode 轮廓的查找方式
○RETR_EXTERNAL 查找最外层轮廓(不查找内层轮廓)
在hierarchy层级结构中,每一个轮廓只有前后轮廓的索引,没有父轮廓与子轮    廓的索引。
○RETR_LIST 查找所有轮廓(包括外层和内层轮廓)
在hierarchy层级结构中,每一个轮廓只有前后轮廓的索引,没有父轮廓与子轮    廓的索引。
○RETR_CCOMP 查找所有轮廓(包括外层和内层轮廓)
在hierarchy层级结构中,每一个轮廓只有前后轮廓的索引,有父轮廓与子轮        廓的索引,轮廓为两层结构。
○RETR_TREE (包括外层和内层轮廓)
在hierarchy层级结构中,每一个轮廓只有前后轮廓的索引,有父轮廓与子轮        廓的索引,轮廓为树状结构。
●method 轮廓的近似办法
○CHAIN_APPROX_NONE 存储所有轮廓点
○CHAIN_APPROX_SIMPLE 只存储有用的点
比如轮廓中的直线,则只存储起点和终点;比如轮廓是四边形,则只存储四个顶点。
○CHAIN_APPROX_TC89_L1 使用Teh_Chin链进行轮廓逼近
精度高(优化后的点更少),使用少。

1.2 绘制轮廓

使用绘制轮廓的方式把上一步的寻找的轮廓点连接在一起进行绘制,这一步绘制的是前景轮廓。

  • contourIdx 轮廓编号

        -1表示绘制所有轮廓

  • color 轮廓颜色
  • thickness 绘制的线宽

1.3 轮廓检测算法 Suzuki

内容复杂,了解即可

2 代码实践

原图3.jpg

import cv2if __name__ == '__main__':# 1. 图片输入path = '3.jpg'image_np = cv2.imread(path)# 2. 灰度化image_np_gray = cv2.cvtColor(image_np,cv2.COLOR_BGR2GRAY)# 3. 二值化ret, image_np_thresh = cv2.threshold(image_np_gray,127,255,cv2.THRESH_BINARY_INV + cv2.THRESH_OTSU)print(ret)# 4. 寻找轮廓contours, hierarchy = cv2.findContours(image_np_thresh,  # 二值化之后的图像cv2.RETR_EXTERNAL,  # 查找方式cv2.CHAIN_APPROX_SIMPLE  # 近似办法)print(len(contours))for i in contours:print(i.shape)"""cv2.findContours(): 查找图像中的轮廓参数详解:image_np_thresh: 二值化图像,必须是8位单通道图像cv2.RETR_EXTERNAL: 轮廓检索模式,只检测最外层轮廓cv2.CHAIN_APPROX_SIMPLE: 轮廓近似方法,压缩水平、垂直和对角线段,只保留它们的端点返回值:contours: 找到的轮廓列表,每个轮廓是一个点集hierarchy: 轮廓的层次信息(此代码中未使用)print(len(contours)): 打印找到的轮廓数量for i in contours: print(i.shape): 遍历所有轮廓并打印每个轮廓的形状(即包含多少个点)"""# 5. 绘制轮廓image_np = cv2.drawContours(image_np,  # 在哪个图上绘制contours,  # 轮廓数据列表contourIdx=-1,  # 绘制轮廓的id,-1表示全绘制color=(0, 0, 255),  # 绘制的颜色thickness=2  # 线宽)"""cv2.drawContours(): 在图像上绘制轮廓参数详解:image_np: 要绘制轮廓的目标图像(会在原图上直接修改)    contours: 要绘制的轮廓列表    contourIdx=-1: 指定要绘制哪个轮廓,-1表示绘制所有轮廓    color=(0, 0, 255): 轮廓颜色,BGR格式(这里是红色)    thickness=2: 轮廓线宽度"""# 6. 图片输出cv2.imshow('image_np', image_np)cv2.waitKey(0)cv2.imwrite('image.png', image_np)# 读取图像 → 2. 转换为灰度图 → 3. 二值化处理 → 4. 查找轮廓 → 5. 绘制轮廓 → 6. 显示结果

运行后图片:image.png

练习:绘制图像的所有轮廓,并使用不同的颜色标识。

图片:3.jpg

"""绘制图像的所有轮廓,并使用不同的颜色标识。"""
import cv2
import numpy as npif __name__ == '__main__':# 1.图片输入path = '3.jpg'  # 图像文件路径image_np = cv2.imread(path)  # 读取图像为NumPy数组(BGR格式)# # 2. 灰度化# image_np_gray = cv2.cvtColor(#     image_np,#     cv2.COLOR_BGR2GRAY# )# cv2.imshow('image_np_gray', image_np_gray)# # 3. 二值化# ret, image_np_thresh = cv2.threshold(#     image_np_gray,#     192,  # 191/192为青色和粉色的临界值#     255,#     cv2.THRESH_BINARY_INV# )# 使用灰度化、二值化后,颜色浅的 无法选中# 2.HSV空间转换hsv_image_np = cv2.cvtColor(image_np, cv2.COLOR_BGR2HSV)  # BGR转HSV# 3. 制作掩膜# 定义绿色范围/背景图掩膜green_low = np.array([35, 43, 46])  # 红色下限(Hmin, Smin, Vmin)green_high = np.array([77, 255, 255])  # 红色上限(Hmax, Smax, Vmax)mask1 = cv2.inRange(  # 创建掩膜hsv_image_np,  # 基于哪个图像, 输入HSV图像green_low,  # 颜色下限green_high  # 颜色上限)# 合并掩膜mask_image_np = cv2.bitwise_or(mask1, mask1)  # 红cv2.imshow('mask_image_np4', mask_image_np)# 4.反阈值二值化取到前景图片ret, image_np_thresh = cv2.threshold(mask_image_np,  # 输入的灰度图像127,  # 阈值(127),但使用OTSU方法时会自动计算最佳阈值255,  # 最大值(255),当像素值超过阈值时赋予的值cv2.THRESH_BINARY_INV + cv2.THRESH_OTSU)cv2.imshow('image_np_thresh', image_np_thresh)# 5. 寻找轮廓contours, hierarchy = cv2.findContours(image_np_thresh,  # 二值化之后的图像cv2.RETR_LIST,  # 查找方式:查找所有轮廓(包括外层和内层轮廓)cv2.CHAIN_APPROX_SIMPLE  # 近似办法)print(len(contours))for i in contours:print(i.shape)# 定义不同颜色用于绘制不同轮廓colors = [(0, 0, 0),  # 黑色(255, 255, 0),  # 青色(255, 0, 0),  # 蓝色(255, 0, 255),  # 粉色]# 遍历所有找到的轮廓for i, cnt in enumerate(contours):# enumerate()函数返回的是(index, value)元组对,所以不能直接将循环变量用于索引# 计算当前轮廓的面积,过滤掉太小的轮廓(可能是噪声)area = cv2.contourArea(cnt)if area < 197:  # 忽略面积小于100像素的轮廓continue# 选择颜色(循环使用预定义的颜色)# color = colors[i]  # 当i≥4时会报IndexErrorcolor = colors[i % len(colors)]  # 0%4=0...5%4=1(循环)# 6. 绘制轮廓image_np = cv2.drawContours(image_np,  # 在哪个图上绘制contours,  # 轮廓数据列表contourIdx=i,  # 绘制轮廓的id,-1表示全绘制,从0开始代表下标color=color,  # 使用选择的颜色thickness=2  # 线宽)# 6. 图片输出cv2.imshow('image_np', image_np)cv2.waitKey(0)cv2.imwrite('123.png', image_np)

运行结果:123.png

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

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

相关文章

Linux shell getopts 解析命令行参数

Linux shell getopts 解析命令行参数getopts语法 getopts 选项字符串 名称 [ 参数 ...]示例1&#xff08;有前置冒号&#xff09;: while getopts ":hdo:" optname; do ...... done示例1&#xff08;无前置冒号&#xff09; while getopts "hdo:" optname…

DeepInteraction++基于多模态交互的自动驾驶感知与规划框架

DeepInteraction++基于多模态交互的自动驾驶感知与规划框架 1 论文核心概念 DeepInteraction++ 提出了一种名为"模态交互"(modality interaction)的新策略,用于自动驾驶中的多模态(LiDAR 和相机)感知任务。其核心思想是不将多模态信息融合为单一表示,而是分别…

忆联参与制定消费级SSD团体标准正式出版! 以“高可靠”引领行业提质增效与用户体验升级

引言​在AIPC爆发、数据价值凸显的当下&#xff0c;存储设备已超越简单容器&#xff0c;成为智能体验基石&#xff0c;其性能与可靠性直接关乎用户效率与资产安全。然而&#xff0c;消费级SSD长期缺乏统一权威的可靠性标准&#xff0c;使厂商缺乏质量对标依据&#xff0c;用户亦…

微服务搭建(SpringBoot + Dubbo + Nacos)

1.项目接口2. 编辑pom.xml和application.yml文件2.1父工程pom.xml<?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0" xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:s…

android中常见布局及其约束

0 布局的定义 布局可以理解为一种​​容器​​&#xff0c;用于​​组织与排列界面上的控件​​。 布局是一个相框&#xff0c;控件就是你要展示的照片。• 你&#xff08;布局规则&#xff09;决定这些照片怎么排列&#xff1a;是从上到下整齐放&#xff08;LinearLayout&am…

Rust语言能干什么

Rust 语言的应用范围非常广&#xff0c;几乎覆盖了现代软件开发的全部领域。它最初以“系统级语言”身份出道&#xff0c;但现在已经远远超出了这个范畴。下面我从几个关键方向给你梳理一下&#xff0c;Rust 到底能干什么&#xff0c;以及为什么在这些领域它特别有优势。 1. 系…

只需一个设置就可以解决Microsoft Edge浏览器打不开网页的问题

Microsoft Edge是一款功能强大的网络浏览器&#xff0c;预装在Windows 10、11系统中。通过这个简单易懂的教程&#xff0c;学习如何修复Microsoft Edge浏览器打不开的问题。1、打开计算机找到C盘&#xff0c;双击打开&#xff1a;2、打开【用户】➜【Admin】➜【AppData】➜【L…

AI 应用 图文 解说 (二) -- 百度智能云 ASR LIM TTS 语音AI助手源码

文章的目的为了记录AI应用学习的经历&#xff0c;降低AI的入门难度。同时记录开发流程和要点有些记忆模糊&#xff0c;防止忘记。也希望可以给看到文章的朋友带来一些收获。 相关链接&#xff1a; AI 应用 图文 解说 (一) -- 百度智能云 实现 语音 聊天-CSDN博客 AI 应用 图文 …

计算机Python毕业设计推荐:基于Django的博客网站设计与实现【python/大数据/深度学习/机器学习定制】

精彩专栏推荐订阅&#xff1a;在下方主页&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb; &#x1f496;&#x1f525;作者主页&#xff1a;计算机毕设木哥&#x1f525; &#x1f496; 文章目录 一、项目介绍二、…

当 AI 开始 “筛选” 信息:算法偏见会加剧认知鸿沟吗?如何构建公平的 AI 生态?

AI 筛选信息的现状与原理​在信息爆炸的时代&#xff0c;AI 筛选信息已成为各领域不可或缺的关键技术。在社交媒体平台上&#xff0c;如抖音、小红书等&#xff0c;AI 根据用户的点赞、评论、浏览历史等数据&#xff0c;精准推送用户可能感兴趣的内容&#xff0c;极大提升了用户…

2023年IEEE IOTJ SCI1区TOP,动态环境下无人机目标覆盖任务路径规划,深度解析+性能实测

目录1.摘要2.问题模型3.算法设计4.结果展示5.参考文献6.代码获取7.算法辅导应用定制读者交流1.摘要 无人机&#xff08;UAV&#xff09;作为物联网应用的重要工具&#xff0c;正广泛应用于智能农业监测、智能交通监测等领域&#xff0c;并逐渐成为国内外研究热点。然而&#x…

计算机视觉(四):二值化

二值化&#xff0c;就是将图像从彩色或灰度模式转换为只有两种颜色&#xff08;通常是黑色和白色&#xff09;的模式。这个过程的本质是设定一个阈值 (Threshold)&#xff0c;将图像中所有像素的灰度值与这个阈值进行比较。 基本原理 二值化的核心原理非常简单&#xff1a; 灰度…

(二)设计模式(Command)

文章目录项目地址一、设计模式1.1 Command Design1. 创建命令接口2. 创建支付的Command类3. CommandScheduler4. 使用1.2 Chain of Responsibility1. 接口创建2. 审批人3. 发起审批1.3 State Pattern1. 创建简单的状态机定义动作和状态状态机使用状态机1.x Iterator1.x Observe…

现代C++性能陷阱:std::function的成本、异常处理的真实开销

1. std::function 的成本 std::function 是一个通用的、类型擦除的函数包装器&#xff0c;它非常方便&#xff0c;可以存储和调用任何可调用对象&#xff08;函数、lambda、函数对象、bind表达式等&#xff09;。然而&#xff0c;这种灵活性是有代价的。 主要成本来源&#xff…

基于Spark的白酒行业数据分析与可视化系统的设计与实现

文章目录有需要本项目的代码或文档以及全部资源&#xff0c;或者部署调试可以私信博主项目介绍一、项目背景与研究意义二、系统整体架构三、系统功能设计四、应用场景与价值五、项目特色与创新点六、总结与展望每文一语有需要本项目的代码或文档以及全部资源&#xff0c;或者部…

织梦会员中心模板调用某个栏目名和栏目下文档的办法

大家在用到织梦dedecms时候&#xff0c;需要在会员中心模板调用栏目的名称和链接&#xff0c;还有某个栏目下的文档要怎么操作呢&#xff1f; 我们都知道&#xff0c;在会员中心模板&#xff0c;直接用dede:type或者dede:arclist标签是不行的&#xff0c;在会员中心调用只能用p…

区块链的法律定位:技术、工具还是资产?

高鹏律师首席数据官&#xff0c;数字经济团队创作AI辅助当我们谈论区块链时&#xff0c;我们在谈论什么&#xff1f;是那串不可篡改的哈希值&#xff0c;是去中心化的信仰图腾&#xff0c;还是藏在代码背后的权利密码&#xff1f;今天&#xff0c;我们不聊技术迭代的炫酷&#…

LeetCode每日一题,2025-8-31

dfs搜索解数独&#xff0c;重点是如何判断当前这位置的board[x][y]是否可以填数字num public class T37 {public static void main(String[] args) {char[][] board new char[][]{{5, 3, ., ., 7, ., ., ., .},{6, ., ., 1, 9, 5, ., ., .},{., 9, 8, ., ., ., ., 6, .},{8, …

Linux 进程信号学习笔记:从概念到实操

一、Linux 信号基本概念1.1 生活角度理解信号我们可以把进程比作等待快递的人&#xff0c;信号就像快递&#xff1a;识别信号&#xff1a;就像我们知道快递来了该 怎么处理&#xff0c;进程对信号的识别是内核程序员预先编写的内置特性&#xff0c;即使信号没产生&#xff0c;进…

解决多种类潮湿敏感元器件的多温度、多时长的排潮烘干

铠德科技ESD烘箱针对复杂电路产品的排潮烘干需求&#xff0c;可通过以下技术路径实现多品类元器件的高效兼容处理&#xff1a;多温区独立控制系统采用蜂窝式加热模块阵列&#xff0c;每个0.6m独立温区可设置1℃精度支持同时运行3种不同温度曲线&#xff08;典型值&#xff1a;8…