opencv中边缘提取、孔洞填充以及阈值分割的实际应用

2023-10-27 23:50

本文主要是介绍opencv中边缘提取、孔洞填充以及阈值分割的实际应用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1、应用中使用的opencv接口

1.1 阈值分割(threshold)

      threshold 函数参数介绍

 double threshold( InputArray src, OutputArray dst,double thresh, double maxval, int type );

      第一个参数为输入的图像,Mat类型的即可。

      第二个参数为输出图像,且和输入图像有同等大小和类型

      第三个参数为设定阙值的具体值

      第四个参数 maxval是当第五个参数类型为CV_THRESH_BINARY和CV_THRESH_BINARY_INV是的最大值

1.2边缘提取

void Canny(Mat image, Mat edges,double threshold1,double threshold2, int aperture_size=3);

   

1.3孔洞填充(findContours)

void findContours( InputOutputArray image, OutputArrayOfArrays contours,OutputArray hierarchy, int mode,int method, Point offset=Point())

使用findContours会同步使用findContours()与边缘检测算法

 

2、待分割图片

带分割图片如下,需要将图片中的小章鱼都分割出来

对图片的处理是先将图片进行颜色空间转换,有RGB转换成为HSV,再基于HSV的图片进行分割;下面三张图分别是进行颜色空间转换、边缘提取、以及孔洞填充后的效果,因为轮廓选择的时候不理想,所以最终分割的效果不好。后面会根据轮廓提取后的效果进一步分割;

以上分割效果不好,后期的处理更偏向于像素的处理,分析样本中蓝色的像素值,把蓝色的区域去掉,代码示例如下:

#include <iostream>
#include <opencv2/opencv.hpp>using namespace std;
using namespace cv;void fillholds(Mat &src, Mat &dst, Mat &sourceImage)
{Mat m_with_border;copyMakeBorder(src, m_with_border, 1, 1, 1, 1, BORDER_CONSTANT, Scalar());copyMakeBorder(sourceImage, sourceImage, 1, 1, 1, 1, BORDER_CONSTANT, Scalar());vector<vector<Point>> contours;vector<Vec4i> hierarchy;// 提取边缘坐标findContours(m_with_border, contours, hierarchy, CV_RETR_TREE, CV_CHAIN_APPROX_NONE);Mat cvImg;sourceImage.copyTo(cvImg);if (!contours.empty()){for (unsigned int idx = 0; idx < contours.size(); idx++){drawContours(cvImg, contours, idx, Scalar::all(0), CV_FILLED, 8);}}Mat resImage = cvImg(Rect(1, 1, m_with_border.cols - 2, m_with_border.rows - 2));//使用孔洞填充的时候部分区域提取异常,对提取异常的区域重新Mat mask = resImage(Rect(0, 0, 230, resImage.rows));for (int i = 0; i < mask.rows; i++){uchar *ptR = resImage.ptr<uchar>(i);uchar *ptS = sourceImage.ptr<uchar>(i);for (int j = 0; j < mask.cols; j++){// 还处理分割失败已经边缘黑色区域if (ptR[j * 3 + 2] == 0 && ptR[j * 3 + 1] == 0 && ptR[j * 3] == 0){ptR[j * 3] = ptS[j * 3];ptR[j * 3 + 1] = ptS[j * 3 + 1];ptR[j * 3 + 2] = ptS[j * 3 + 2];}// 再对最后一个章鱼进行分割if (ptR[j * 3 + 2] > 150){ptR[j * 3] = 0;ptR[j * 3 + 1] = 0;ptR[j * 3 + 2] = 0;}}}// 对部分比分割不好的进行形态学变换Mat ele = getStructuringElement(MORPH_RECT, Size(3, 3));morphologyEx(resImage, resImage, MORPH_ERODE, ele);dst = resImage;}int main()
{string sPath = "C:\\Users\\Administrator\\Desktop\\img\\segment.png";Mat src = imread(sPath);Mat srcimg;src.copyTo(srcimg);Mat hsvSrc;cvtColor(src, hsvSrc, CV_BGR2HSV);Mat showImg(src.rows * 2, src.cols * 2, CV_8UC3);vector<Mat> vImg;split(hsvSrc, vImg);Mat img1 = vImg[0];Mat cannyBoredr;Canny(vImg[0], cannyBoredr, 60, 180, 3);Mat mask;threshold(cannyBoredr, mask, 0, 255, THRESH_OTSU);Mat dst1;fillholds(mask, dst1, srcimg);// 使用直接减的方式会将上一步形态学变换的部分区域显示出来,达不到最好的效果Mat resultImage = src - dst1;// 使用二次分割的方式再进行一次提取Mat segImg(dst1.rows, dst1.cols,CV_8UC3);segImg = Scalar(0);for (int i = 0; i < segImg.rows; i++){uchar *ptR = segImg.ptr<uchar>(i);uchar *ptS = src.ptr<uchar>(i);uchar *ptB = dst1.ptr<uchar>(i);for (int j = 0; j < segImg.cols; j++){if (ptB[j * 3]	< 50){ptR[j * 3] = ptS[j * 3];ptR[j * 3 + 1] = ptS[j * 3 + 1];ptR[j * 3 + 2] = ptS[j * 3 + 2];}}}system("pause");return 0;
}

代码处理中的中间图如下,第一张图是轮廓提取后的效果图,可以明显看见,左侧的小章鱼分割失败,此时就需要对这个小章鱼进行额外提取;第二张图是根据像素值分割后的掩码结果;由于掩码图经过膨胀炒作,所以直接使用原图减mask图就出现第三张图的效果;最后一张是基于像素重新处理后的图片,分割效果较好。

上述只是介绍了边缘提取、孔洞填充的使用,代码可读性比较差,后续重构;

https://mp.csdn.net/editor/html?spm=1010.2135.3001.5352

这篇关于opencv中边缘提取、孔洞填充以及阈值分割的实际应用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/289265

相关文章

使用Java填充Word模板的操作指南

《使用Java填充Word模板的操作指南》本文介绍了Java填充Word模板的实现方法,包括文本、列表和复选框的填充,首先通过Word域功能设置模板变量,然后使用poi-tl、aspose-words... 目录前言一、设置word模板普通字段列表字段复选框二、代码1. 引入POM2. 模板放入项目3.代码

利用Python操作Word文档页码的实际应用

《利用Python操作Word文档页码的实际应用》在撰写长篇文档时,经常需要将文档分成多个节,每个节都需要单独的页码,下面:本文主要介绍利用Python操作Word文档页码的相关资料,文中通过代码... 目录需求:文档详情:要求:该程序的功能是:总结需求:一次性处理24个文档的页码。文档详情:1、每个

Java中的分布式系统开发基于 Zookeeper 与 Dubbo 的应用案例解析

《Java中的分布式系统开发基于Zookeeper与Dubbo的应用案例解析》本文将通过实际案例,带你走进基于Zookeeper与Dubbo的分布式系统开发,本文通过实例代码给大家介绍的非常详... 目录Java 中的分布式系统开发基于 Zookeeper 与 Dubbo 的应用案例一、分布式系统中的挑战二

Java 缓存框架 Caffeine 应用场景解析

《Java缓存框架Caffeine应用场景解析》文章介绍Caffeine作为高性能Java本地缓存框架,基于W-TinyLFU算法,支持异步加载、灵活过期策略、内存安全机制及统计监控,重点解析其... 目录一、Caffeine 简介1. 框架概述1.1 Caffeine的核心优势二、Caffeine 基础2

使用Node.js和PostgreSQL构建数据库应用

《使用Node.js和PostgreSQL构建数据库应用》PostgreSQL是一个功能强大的开源关系型数据库,而Node.js是构建高效网络应用的理想平台,结合这两个技术,我们可以创建出色的数据驱动... 目录初始化项目与安装依赖建立数据库连接执行CRUD操作查询数据插入数据更新数据删除数据完整示例与最佳

MyBatis Plus实现时间字段自动填充的完整方案

《MyBatisPlus实现时间字段自动填充的完整方案》在日常开发中,我们经常需要记录数据的创建时间和更新时间,传统的做法是在每次插入或更新操作时手动设置这些时间字段,这种方式不仅繁琐,还容易遗漏,... 目录前言解决目标技术栈实现步骤1. 实体类注解配置2. 创建元数据处理器3. 服务层代码优化填充机制详

PHP应用中处理限流和API节流的最佳实践

《PHP应用中处理限流和API节流的最佳实践》限流和API节流对于确保Web应用程序的可靠性、安全性和可扩展性至关重要,本文将详细介绍PHP应用中处理限流和API节流的最佳实践,下面就来和小编一起学习... 目录限流的重要性在 php 中实施限流的最佳实践使用集中式存储进行状态管理(如 Redis)采用滑动

深入浅出Spring中的@Autowired自动注入的工作原理及实践应用

《深入浅出Spring中的@Autowired自动注入的工作原理及实践应用》在Spring框架的学习旅程中,@Autowired无疑是一个高频出现却又让初学者头疼的注解,它看似简单,却蕴含着Sprin... 目录深入浅出Spring中的@Autowired:自动注入的奥秘什么是依赖注入?@Autowired

Python从Word文档中提取图片并生成PPT的操作代码

《Python从Word文档中提取图片并生成PPT的操作代码》在日常办公场景中,我们经常需要从Word文档中提取图片,并将这些图片整理到PowerPoint幻灯片中,手动完成这一任务既耗时又容易出错,... 目录引言背景与需求解决方案概述代码解析代码核心逻辑说明总结引言在日常办公场景中,我们经常需要从 W

Java+AI驱动实现PDF文件数据提取与解析

《Java+AI驱动实现PDF文件数据提取与解析》本文将和大家分享一套基于AI的体检报告智能评估方案,详细介绍从PDF上传、内容提取到AI分析、数据存储的全流程自动化实现方法,感兴趣的可以了解下... 目录一、核心流程:从上传到评估的完整链路二、第一步:解析 PDF,提取体检报告内容1. 引入依赖2. 封装