一 Contour Finding

    Contours使用 STL-style vector<> 表示,如 vector<cv::Point>, vector<cv::Point2f>。opencv中,使用函数 cv::findContours() 寻找contours, 具体函数定义如下:

    void cv::findContours(cv::InputOutputArray image,

                                       cv::OutputArrayofArrays contours,

                                       cv::OutputArray hierarchy,

                                       int mode,

                                       int method,

                                       cv::Point offset = cv::Point());

     void cv::findContours(cv::InputOutputArray image,

                                       cv::OutputArrayofArrays contours,

                                       int mode,

                                       int method,

                                       cv::Point offset = cv::Point());

    参数 image 为8位单通道输入图像,一般情况下,该图像可能由 cv::threshold(), cv::adaptiveThreshold 生成。当 image 由 cv::Canny() 生成时,cv::findContours() 仅当边缘图像为宽度为1的细区域图像,对于闭合边缘,可以使用内边缘或者外边缘代替边缘图像进行后续分析;但对于非闭合边缘,个人认为 cv::Canny() 生成的边缘图像不适合使用 cv::findContours() 进行查找;替代方案是使用边缘跟踪算法,将 cv::Canny()  生成边缘保存在 vector<cv::Point> 中,然后使用 Contours 相关分析进行更多分析。

    参数 contours 为 vector<vector<cv::Point>>, vector<vector<cv::Point2f>>, 使用 array of arrays 结构可以同时保存多条 contours。

    参数 hierarchy 为 vector<cv::Vec4i>, 每个元素对应一个 contour, 元素中4个整数表示该 contour 与其他 contour 之间的关系,具体关系如下:

        

    其中, c 表示 contour, h 表示  hole.

    opencv 如何生成 contour 的 hierarchy? 个人认为(暂时没有研究 opencv 具体实现),一种可行的思路是使用区域增长,大概流程如下:

    1)将种子点设置在图像边角处,使用背景颜色进行区域增长,区域增长在前景处停止;

    2)使用形态学算子对当前增长后区域进行 dilate 操作,使 c0, c1,... 层级上的 contours 包含在当前增长区域中;

    3)继续使用前景对图像增长,将 h00, h01,..., h10, h11,... 等层级上的 contours 包含在当前增长区域中;

    4)如此循环,即可生成 contour 的  hierarchy 结构。

    要使用 vector<cv::Vec4i> 表示 以上树形结构,每个 cv::Vec4i 元素值含义如下:

    1) index 0: next contour(same level);

    2) index 1: privous contour(same level);

    3) index 2: first child(next level down); 

    4) index 3: parent(next level up);

   参数 mode 表示如何提取 contours, 具体如下:

    1) cv::PETR_EXTERNAL: 仅提取最外层 contour, 在以上图例中,仅提取 c0;

    2) cv::PETR_LIST: 提取所有 contours, 组成链表结构,在图例中为: c010->c001->c000->h01->h00->c0;

    3) cv::PETR_CCOMP: 提取所有 contours, 根据类型组成  hole list 和 contour list, 在图例中为:c010->c001->c000->c0, h01->h00, c0->h01(此处将前两个链表连接起来);

    4) cv::PETR_TREE: 提取所有 contours, 建立树形结构。

    参数 method 对 contour 压缩,以减少数据量,包括:

    1) cv::CHAIN_APPROX_NONE: 不做任何压缩;

    2) cv::CHAIN_APPROX_SIMPLE: 压缩水平,垂直,对角方向上的线段;

    4) cv::CHAIN_APPROX_TC89_L1/cv::CHAIN_APPROX_TC89_KCOS: 使用曲线曲率信息做更复杂的压缩。

    参数 offset 将 contour 平移,在使用ROI进行 contour 提取时, 通过设置不同 offset 可以很方便的将 contour 绘制在原图上。

二  Contour Drawing

    使用 cv::drawContours() 在图像上绘制 contours, 具体定义如下:

    void cv::drawContours(cv::InputOutputArray image,

                                         cv::InputArrayofArrays contours,

                                         int contourIdx,

                                         const cv::Scalar& color,

                                         int thickness = 1, 

                                         int lineType = 8,

                                         cv::InputArray hierarchy = noArray(),

                                         int maxLevel = INT_MAX(),

                                         cv::Point offset = cv::Point());

    参数 image 为需要绘制 contours 图像。

    参数 contours 为 cv::findContours() 生成。

    参数 contourIdx 选择需要绘制的 contour, 如果其值为 -1, 则绘制所有 contours。

    参数 color thickness lineType 分别定义线条颜色,线条宽度,线条类型(4连接 cv::LINE_4, 8连接 cv::LINE_8, 反走样cv::LINE_AA)。

    参数 hierarchy maxLevel 共同控制 contours 层级。

    参数 offset 表示绘制时平移值。

三 Contour Operation

    1 cv::approxPolyDP() 对  contour 进行多边形近似,具体定义如下:

    void cv::approxPolyDP(cv::InputArray curve, cv::OutputArray approxCurve, double epsilon, bool closed);

    参数 curve 可以是 vector<cv::Point>, vector<cv::Point2f>  或者 arrays of size N*2, arrays of size N*1 with 2 channels。

    参数 epsilon 表示近似精度,一般通过 contour 长度的百分比计算得出。

    参数 closed 表示 contour 是否闭合。

    cv::approxPolyDP() 函数使用 Douglas-Peucker approximation,基本思路如下:

    1)在 contour 上寻找距离最大的两个点,将 contour 一分为二, 并连接两点构成线段S;

    2)  在两个半边缘上分别寻找到到线段S上的最远点,将半边缘一分为二,连接以上四个点形成四边形;

   3)继续寻找到四边形各条边上的最远点,构成多边形,知道最远点到对应边上距离小于epsilon停止。

   2 double cv::arcLength(cv::InputArrray points, bool closed) 求 contour 长度。

   3 double cv::contourArea(cv::InputArray points, bool oriented = false) 使用格林公式求 contour 所围成的面积,oriented = true 时返回带符号面积值,该函数对复杂区域(如自交区域)将返回错误结果。

   4 cv::Rect cv::boundingRect(cv::InputArray points) 求 contour 所围成的矩形(无旋转矩形)。

   5 cv::RotateRect cv::minAreaRect(cv::InputArray points) 求 contour 所围成的最小矩形(旋转矩形)。

   6 void cv::minEnclosingCircle(cv::InputArray points, cv::Point2f& center, float& radius) 求 contour 所围成的最小圆形。

   7 cv::RotateRect cv::fitEllipse(cv::InputArray points) 使用 contour 拟合椭圆(使用最小化代价函数)。

   8 cv::fitLine() 拟合直线,具体定义如下:

    void cv::fitLine(cv::InputArray points, cv::OutputArray line, int distType, double param, double reps, double aeps);

    参数 points 接受二维或者三维点,拟合成二维平面直线或者三维空间直线。

    参数 line 为 vec4f 或者 vec6f, 前两个(或三个)元素给出直线方向,后两个(或三个)给出直线上一个点。

    参数  distType 表示距离度量范数,一般使用cv::DIST_L2, cv::DIST_L1。

    参数 reps aeps 表示直线方向与直线点的计算精度,一般赋值为 .01 即可。

   9 void cv::convexHull(cv::InputArray points, cv::OutputArray hull, bool clockwise = false, bool returnPoints = true) 计算 contour 凸包,凸包点集计算比较简单。

   10 bool cv::isContourConvex(cv::InputArray contour) 判断  contour  凸性。

   11 double cv::pointPolygonTest(cv::InputArray contour, cv::Point2f pt, bool measureDist) 判断点是否在 contour 中,如果 measureDist  = true, 返回点到最近边缘距离。

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

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

相关文章

网络安全初级

1、docker并配置代理 &#xff08;1&#xff09;在Ubuntu中安装docker apt-get install docker.io docker-compose &#xff08;2&#xff09;安装完成后&#xff0c;进入/etc/systemd/system/docker.service.d/http-proxy.conf配置文件下进行代理的配置&#xff0c;配置如图…

JetBrains IDE 性能优化指南:idea.vmoptions 核心参数解析与配置建议

文章目录深入解析 JetBrains IDE 的 VM 选项&#xff1a;idea.vmoptions 参数详解一、内存与垃圾回收配置二、诊断与错误处理三、运行时优化参数四、模块系统与反射控制五、特殊参数说明六、配置建议指南深入解析 JetBrains IDE 的 VM 选项&#xff1a;idea.vmoptions 参数详解…

Datawhale AI夏令营 《基于带货视频评论的用户洞察挑战赛》Part .1.

1. 赛题 参赛者需要构建端到端的评论分析系统&#xff0c;完成三大核心任务&#xff1a; 商品识别 输入&#xff1a;视频描述文本(video_desc)和标签(video_tags)输出&#xff1a;精准识别推广商品名称(Xfaiyx Smart Translator/Recorder) 多维情感分析 维度&#xff1a;情感倾…

【博文汇项目全维度测试报告:功能与自动化双轨验证】

&#x1f308;个人主页: Aileen_0v0 &#x1f525;热门专栏: 华为鸿蒙系统学习|计算机网络|数据结构与算法 ​&#x1f4ab;个人格言:“没有罗马,那就自己创造罗马~” 文章目录 项目背景:项目背景与意义&#xff1a;项目概述已实现的主要功能包括&#xff1a;当前系统存在的不足…

Java陷阱之assert关键字详解

Assert.isTrue()方法用于断言条件是否为真&#xff0c;如果条件不满足&#xff08;即为false&#xff09;&#xff0c;则会抛出IllegalArgumentException&#xff0c;并附带预设的错误信息。在示例中&#xff0c;当1.23不小于2.23时&#xff0c;方法抛出了异常&#xff0c;显示…

mysql 散记:innodb引擎和memory引擎对比 sql语句少用函数 事务与长事务

文章目录innodb引擎和memory引擎对比对比sql 语句&#xff1a;尽可能不使用函数条件隐式转换隐式类型转换隐式字符编码转换补充问题事务与长事务ACIDread viewMVCC 一致性视图当前读view 虚拟表长事务的影响与排查影响排查方法预防innodb引擎和memory引擎对比 innodb引擎是索引…

APK安装器(安卓端)一键解除VX限制!轻松安装各种手机应用

VX为了防止恶意软件通过平台传播&#xff0c;保障用户设备安全&#xff0c;会把通过VX发送的 APK 文件自动改成 “apk.1” 格式&#xff0c;这样就能减少恶意软件传播的风险。我平时推荐安卓软件的时候&#xff0c;有朋友反馈说&#xff0c;文件发到VX里就变成 “apk.1” 了&am…

Debian:从GNOME切换到Xfce

最近为20年前的T43重新安装了Debian系统&#xff0c;但使用的gnome桌面太卡了。于是换成轻量级的Xfce系统。 1.安装Xfce sudo apt update sudo apt install xfce4 xfce4-goodies命令中xfce4 是Xfce桌面环境的核心组件&#xff0c;xfce4-goodies 是一些额外的工具和插件&#xf…

徐州服务器租用:BGP线路的特点有哪些?

BGP的中文全称为边界网关协议&#xff0c;是指一种运行在传输控制协议上的自治系统路由协议&#xff0c;主要的功能就是可以实时控制路由的传播&#xff0c;同时能够帮助用户选择更合适的路由线路&#xff0c;保证网络能够稳定的运行&#xff0c;不会轻易出现网络卡顿或故障的状…

Java使用OSHI获取服务器信息

OSHI可以获取系统信息&#xff08;CPU、内存、磁盘、网络等&#xff09;&#xff0c;纯Java实现&#xff08;通过JNA访问本地API&#xff0c;无需安装本地库&#xff09;&#xff0c;跨平台支持。引入依赖<dependency><groupId>com.github.oshi</groupId><…

企业数字化资产管理安全、成本、协作困局难解?

在数字化浪潮席卷全球的今天&#xff0c;3D技术已成为驱动影视动画、工业设计、建筑可视化等领域创新的核心动力。然而&#xff0c;随着3D资产规模呈指数级增长&#xff0c;企业正面临前所未有的管理挑战&#xff1a;海量模型存储混乱、版本迭代难以追溯、团队协作效率低下、知…

力扣面试150题--组合总和

Day 72 题目描述&#xff08;终于理顺回溯了&#xff09;思路 这里还是基于模板来说明代码思路void backtracking(参数) {if (终止条件) {存放结果;return;}for (选择 : 本层集合中的元素) {处理节点;backtracking(路径, 选择列表); // 递归撤销处理; // 回溯} }对于主要函数的…

多客户端-服务器(select,poll)

多客户端-服务器结构总结一、普通CS架构的局限性核心问题&#xff1a;单线程中accept&#xff08;阻塞等待连接&#xff09;与read&#xff08;阻塞读取数据&#xff09;函数互相干扰&#xff0c;无法同时处理多客户端。本质原因&#xff1a;阻塞型函数需独立执行&#xff0c;若…

如何使用postman做接口测试?

&#x1f345; 点击文末小卡片 &#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 常用的接口测试工具主要有以下几种&#xff1a;Postman: 简单方便的接口调试工具&#xff0c;便于分享和协作。具有接口调试&#xff0c;接口集管理&#xff0c…

新型网络架构设计助力智慧医疗降本增效

随着智慧医疗的快速发展,越来越多的医院开始布局“互联网+医疗”服务,通过数字化手段提升医疗服务效率。然而,如何构建一个既稳定可靠又具备灵活扩展能力的医疗网络,成为医院数字化转型中的关键问题。本文以某智慧医疗项目为例,探讨传统网络与SD-WAN结合的最佳实践。 背景…

一文读懂现代卷积神经网络—使用块的网络(VGG)

目录 什么是使用块的网络&#xff08;VGG&#xff09;&#xff1f; 一、VGG 的核心思想&#xff1a;用块&#xff08;Block&#xff09;构建网络 二、VGG 的网络结构 三、VGG 的优势 1. 结构简洁统一 2. 强大的特征表达能力 3. 小卷积核的计算效率 4. 良好的迁移学习性…

Linux的相关学习

linux 1.文件权限怎么修改 chmod [权限模式] [文件或目录]1、**数字模式&#xff08;八进制&#xff09;**&#xff1a; chmod 755 myfile.sh # 所有者&#xff1a;rwx (7)&#xff0c;组&#xff1a;r-x (5)&#xff0c;其他用户&#xff1a;r-x (5) 7 rwx&#xff08;读写…

Kotlin集合接口

Kotlin 集合概述 Kotlin 集合提供了对数据进行各种操作的便捷方式。它们实现了接口&#xff0c;因此可以操作不同类型的数据。例如&#xff0c;你可以编写一个函数&#xff0c;同时打印 Set 和 List 的所有元素。我们来看看这是如何实现的。Iterable 接口 我们已经知道&#xf…

Git 常用操作与注意事项全攻略

1. 基本配置 git config --global user.name "你的名字" git config --global user.email "你的邮箱" git config --list # 查看当前配置建议全局配置用户名和邮箱&#xff0c;否则提交记录可能不规范2.仓库操作 初始化本地仓库 git init只在新建项目时使…

STM32-第五节-TIM定时器-1(定时器中断)

一、定时器原理&#xff1a;1.介绍&#xff1a;对指定输入时钟进行计数&#xff0c;并在计数值达到设定值时触发中断。分类&#xff1a;基本定时器&#xff0c;通用定时器&#xff0c;高级定时器频率&#xff1a;72MHZ2.框图&#xff1a; &#xff08;1&#xff09;基本定时器&…