OPENCV——形态学基础之膨胀、腐蚀
一、 膨胀的原理数学表达式dst(x,y) dilate(src(x,y)) max(x,y)src(xx,yy)膨胀是图像形态学的基本功能之一膨胀顾名思义就是求图像的局部最大值操作它的数学表达式是dst(x,y) dilate(src(x,y)) max(x,y)src(xx,yy)。若从数学的角度来看无论是膨胀还是腐蚀实际上就是把图像跟核进行卷积(卷积是通过两个函数f和g生成第三个函数的一种数学运算它的本质就是微积分的转换积分转换的数学公式(f * g)(t) ∫f(τ)g(t -τ)dτ)如上图1图像A和形状B进行卷积操作然后形成右边的图像右边的图像就是AB的图像点。所谓的核就是指任意的形状或者大小在多数情况下核是一个小的中间带的正方形或者圆形。膨胀本质上就是把图像与核进行卷积操作然后计算出卷积区域的最大点并把最大的值赋值给指定的像素(如上图)。操作完成之后图像就会更加明亮(如下图)。上图就是膨胀前和膨胀后图像的对比。从这张图我们可以看出来右边经过dilate膨胀操作后整个图像更加的明亮和粗糙。二、膨胀的API讲解2.1.dilate的API在OPENCV中有一个专门的API去处理图像的膨胀这个API就是dilatevoid dilate( InputArray src, OutputArray dst, InputArray kernel, Point anchor Point(-1, -1), int iterations 1, int borderType BORDER_CONSTANT, const Scalar borderValue morphologyDefaultBorderValue() );第一个参数src的类型是InputArray它指的是输入图像它可以是Mat类的数据。图像的通道数可以是任意数但是图像的深度一般是CV_8UCV_16UCV_16SCV_32FCV_64F第二个参数dst的类型是OutputArray它指的是目标图像值得注意的是输出图像的尺寸、类型要和输入图像是一致的。第三个参数InputArray类型的kernel膨胀操作的核。当这个值为NULL的时候表示使用的核参考点默认是3*3。这个参数通常会配合getStructingElement参数的使用(这个参数的使用下面会详细说到)。第四个参数Point类型的anchor描点的位置默认是(-1,-1)表示中心位置。第五个参数int类型的迭代次数默认是1意思是连续执行几次膨胀1向外扩 1 圈最常用轻微补缺口、消锯齿2向外扩 2 圈效果更强能填更大的洞、接更远的断裂次数越多目标形状失真越严重工程上一般不超过 3 次小技巧2 次 3×3 膨胀 ≈ 1 次 5×5 膨胀效果接近计算量略有差异第六个参数int类型的borderType这个类型用于推断图像外部的边界模式用的最多的是BORDER_DEFAULT下面是常用的几种边框模式(这几种相对比较常用其他的用的很少)BORDER_CONSTANT用指定的像素填充边框BORDER_REPLICATE用已知的边缘像素来填充边框BORDER_WRAP用另一边的像素来补偿填充BORDER_DEFAULT默认模式画边框BORDER_TRANSPANT:用透明的方式画框第七个参数const Scalar类型的borderType一般不用填写因为这个API已经有了默认值morphologyDefaultBorderValue()2.2.getStructingElement的API该函数的作用是返回一个卷积层Mat getStructuringElement( int shape, Size ksize, Point anchor Point(-1, -1) );第一个参数表示内核的形状这里包括了矩形(MORPH_RECT)、交叉形(MORPH_CROSS)、椭圆形(MORPH_ELLIPSE)常用的内核形状是矩形枚举值形状适用场景MORPH_RECT矩形核最通用方方正正膨胀适合方形、规则目标计算最快嵌入式优先选MORPH_ELLIPSE椭圆形核膨胀后边缘更圆润平滑适合圆形、文字、不规则目标效果更自然MORPH_CROSS十字形核只在横竖方向膨胀适合横竖线条类目标第二个参数内核的尺寸格式Size(宽, 高)一般宽高相等且为奇数保证有中心点常用规格Size(3,3)轻微膨胀补小缺口、消锯齿对目标尺寸影响极小90% 场景首选Size(5,5)中等强度膨胀填小孔、接断裂目标会明显变粗更大的核7×7 以上强膨胀会严重改变目标形状非特殊需求不建议用生成核的标准写法// 最常用3×3矩形核通用、速度快 Mat kernel getStructuringElement(MORPH_RECT, Size(3, 3)); // 圆润效果3×3椭圆核适合文字、圆形 Mat kernel_ellipse getStructuringElement(MORPH_ELLIPSE, Size(3, 3));第三个参数锚点的位置默认值Point(-1,-1)表示的是位于中心点三、代码实现图像膨胀功能3.1.用OPENCV实现膨胀功能的大体流程图从上面的流程图我们可以看出来OPENCV实现膨胀功能需要有以下几步分别是imread读取图片、使用cvtColor对图片进行灰度操作、使用getStructingElement获取卷积层、使用dilate对图片进行膨胀、imwrite保存图片3.2.代码#include opencv2/opencv.hpp #include opencv2/dnn.hpp #include opencv2/imgcodecs.hpp #include opencv2/imgproc.hpp #include iostream using namespace cv; using namespace std; int main() { Mat testImage imread(drink.png); if(testImage.empty()) { printf(read testImage failed....\n); return -1; } Mat vertical_structure getStructuringElement(MORPH_RECT, Size(15,15)); dilate(testImage, testImage, vertical_structure); imwrite(dilate_process.jpg, testImage); return 0; }1.读取我们需要处理的图片Mat testImage imread(drink.png); if(testImage.empty()) { printf(read testImage failed....\n); }使用imread读取我们需要处理的图片2.获取卷积层Mat vertical_structure getStructuringElement(MORPH_RECT, Size(15,15));使用getStructuringElement获取卷积层参数设置内核的形状是MORPH_RECT、内核的长度是Size(15,15)3. dilate对图片进行膨胀操作dilate(testImage, testImage, vertical_structure);获取完卷积层后我们就对图片进行dilate膨胀关于dilate的API上面已经详细说到了。最后用imwrite保存图片经过上述处理过后我们来看看原图和处理后的图片效果左边是原图右边是经过膨胀后的图片。我们可以看到右边经过膨胀的图片明显更加明亮和粗糙这是因为dilate就是把原图和卷积层进行重叠整个图像的像素都会变大。四、腐蚀的原理数学表达式dst(x,y) erode(src(x,y)) min(x,y)src(xx,yy)腐蚀也是图像形态学的基本功能之一腐蚀跟膨胀属于反向操作膨胀是把图像图像变大而腐蚀就是把图像变小。腐蚀的原理就是指定一个卷积内核然后原图像和卷积内核进行局部最小值的计算最终获取到一个局部像素点最小的图像如上图1数学表达式是dst(x,y) erode(src(x,y)) min(x,y)src(xx,yy)。经过erode处理后我们来看看两个图片的区别看下图左边是原图右边是经过腐蚀过后的图像。我们可以明显看出来右边的图像字母部分明显比左边的原图像更加细小但是黑暗背景部分会更加大。五、腐蚀的API讲解5.1 erode的函数定义erode是OPENCV实现腐蚀效果的APIvoid erode( InputArray src, OutputArray dst, InputArray kernel, Point anchor Point(-1, -1), int iterations 1, int borderType BORDER_CONSTANT, const Scalar borderValue morphologyDefaultBorderValue() );第一个参数src的类型是InputArray它指的是输入图像它可以是Mat类的数据。图像的通道数可以是任意数但是图像的深度一般是CV_8UCV_16UCV_16SCV_32FCV_64F第二个参数dst的类型是OutputArray它指的是目标图像值得注意的是输出图像的尺寸、类型要和输入图像是一致的。第三个参数InputArray类型的kernel膨胀操作的核。当这个值为NULL的时候表示使用的核参考点默认是3*3。这个参数通常会配合getStructingElement参数的使用(这个参数的使用下面我会详细说到)。第四个参数Point类型的anchor描点的位置默认是(-1,-1)表示中心位置。第五个参数int类型的迭代次数默认是1第六个参数int类型的borderType这个类型用于推断图像外部的边界模式它的默认值是BORDER_DEFAULT第七个参数const Scalar类型的borderType一般不用填写因为这个API已经有了默认值morphologyDefaultBorderValue()5.2 getStructingElement的函数定义getStructingElement的作用是返回一个卷积层CV_EXPORTS_W Mat getStructuringElement(int shape, Size ksize, Point anchor Point(-1,-1));第一个参数表示内核的形状这里包括了矩形(MORPH_RECT)、交叉形(MORPH_CROSS)、椭圆形(MORPH_ELLIPSE)第二个参数内核的尺寸第三个参数锚点的位置默认值Point(-1,-1)表示的是位于中心点六、OPENCV代码实现图像腐蚀功能6.1用OPENCV实现腐蚀功能的大体流程图从上面的流程图我们可以看出来OPENCV实现膨胀功能需要有以下几步分别是imread读取图片、使用getStructingElement获取卷积层、使用erode对图片进行腐蚀、imwrite保存腐蚀后的图片6.2代码#include opencv2/imgcodecs.hpp #include opencv2/highgui.hpp #include opencv2/imgproc.hpp #include iostream using namespace cv; using namespace std; int main() { Mat testImage imread(drink.png); if(testImage.empty()) { printf(could not load image.....\n); return -1; } Mat vertical_structure getStructuringElement(MORPH_RECT, Size(15,15)); erode(testImage, testImage, vertical_structure); imwrite(erode_process.jpg, testImage); return 0; }1.读取我们需要处理的图片Mat testImage imread(drink.png); if(testImage.empty()) { printf(could not load image.....\n); return -1; }使用imread读取我们需要处理的图片2. 获取卷积层Mat vertical_structure getStructuringElement(MORPH_RECT, Size(15,15));使用getStructuringElement获取卷积层参数设置内核的形状是MORPH_RECT、内核的长度是Size(15,15)3. erode对图片进行腐蚀操作erode(testImage, testImage, vertical_structure);获取完卷积层后我们就对图片进行erode腐蚀关于erode的API上面已经详细说到了最后调用imwrite保存图片(这里就省略不写出来)。经过上面处理后我们来看看处理后的图片左边是原图右边是经过腐蚀后的图片。我们可以看到右边经过腐蚀的图片饮料部分图片变小了而整个背景则加深了。

相关新闻

机器学习与模式识别 第一章 机器学习导论 模拟卷及答案

机器学习与模式识别 第一章 机器学习导论 模拟卷及答案

第一章:Introduction to Machine Learning — 单元习题总分:100分 | 建议用时:60分钟 范围:ML定义、AI vs ML、三类问题、学习范式、ML历史、ML生命周期、归纳偏置占位用一、单项选择题(每题2分,共20题&…

2026/6/30 23:01:40阅读更多 →
【射频IC】毫米波CMOS PA设计指北——级间增益匹配

【射频IC】毫米波CMOS PA设计指北——级间增益匹配

前言 级联PA多使用片上变压器进行级间匹配。本文写一写变压器参数如何影响级联放大器的小信号增益,从而方便调参。 级间匹配与输出匹配的区别 在讨论调参方法前,本文先陈述背景电路与基本原理。下图展示了级间匹配等效电路的由来。 (a)驱动级放大器&…

2026/6/30 23:01:40阅读更多 →
E-Hentai下载器完整指南:如何轻松批量下载并打包图片资源

E-Hentai下载器完整指南:如何轻松批量下载并打包图片资源

E-Hentai下载器完整指南:如何轻松批量下载并打包图片资源 【免费下载链接】E-Hentai-Downloader Download E-Hentai archive as zip file 项目地址: https://gitcode.com/gh_mirrors/eh/E-Hentai-Downloader 你是否曾经面对E-Hentai画廊中成百上千的图片&…

2026/6/30 23:01:40阅读更多 →
2026年国内口碑好的电力测功机销售厂家,究竟有哪些值得关注?

2026年国内口碑好的电力测功机销售厂家,究竟有哪些值得关注?

在工业制造、科研等众多领域,电力测功机作为一种重要的测试设备,其性能和质量直接影响着产品的研发和生产。2026年,国内有不少口碑良好的电力测功机销售厂家,其中杭州索川科技有限公司(以下简称索川科技)就…

2026/7/1 0:06:44阅读更多 →
从提示工程到上下文工程:2026年AI开发者的核心技能转换

从提示工程到上下文工程:2026年AI开发者的核心技能转换

# 从提示工程到上下文工程:2026年AI开发者的核心技能转换## 一、背景:Prompt工程的瓶颈已经到来2025年初,当大多数AI开发者还在钻研如何写出“更优美的Prompt”时,一个根本性的认知转变正在顶尖团队中发生。Andrej Karpathy在一次…

2026/7/1 0:06:44阅读更多 →
TwitchDropsMiner:无需观看直播,自动化获取Twitch掉落奖励的终极指南

TwitchDropsMiner:无需观看直播,自动化获取Twitch掉落奖励的终极指南

TwitchDropsMiner:无需观看直播,自动化获取Twitch掉落奖励的终极指南 【免费下载链接】TwitchDropsMiner An app that allows you to AFK mine timed Twitch drops, with automatic drop claiming and channel switching. 项目地址: https://gitcode.c…

2026/7/1 0:06:44阅读更多 →
别再死记硬背了!用‘分界线’思维彻底搞懂C++ set的lower_bound和upper_bound

别再死记硬背了!用‘分界线’思维彻底搞懂C++ set的lower_bound和upper_bound

用‘分界线’思维彻底掌握C set的lower_bound和upper_bound在C标准模板库(STL)中,set容器因其自动排序和快速查找的特性而广受欢迎。然而,许多初学者在使用lower_bound和upper_bound这两个关键方法时,常常陷入死记硬背"大于"或&quo…

2026/7/1 0:06:44阅读更多 →
biliTickerBuy:B站会员购抢票工具的终极指南与实战技巧

biliTickerBuy:B站会员购抢票工具的终极指南与实战技巧

biliTickerBuy:B站会员购抢票工具的终极指南与实战技巧 【免费下载链接】biliTickerBuy b站会员购购票辅助工具 项目地址: https://gitcode.com/GitHub_Trending/bi/biliTickerBuy 在B站会员购抢票的激烈竞争中,手动操作往往难以应对毫秒级的竞争…

2026/7/1 0:06:44阅读更多 →
餐饮老板必看:扫码点餐小程序3步搞定,别再让顾客干等了!

餐饮老板必看:扫码点餐小程序3步搞定,别再让顾客干等了!

目录 第一步:选对模板,省心一半 第二步:打开扫码点餐功能 开启功能按钮 桌台管理与桌码生成 第三步:个性化设计,打造品牌感 调整点餐页面 设置点餐规则 你还在让顾客站着排队点餐吗?2025年&#xff…

2026/7/1 0:01:44阅读更多 →
AI Coding 六个月真实ROI账本:产品经理的血泪教训,研发的冷静忠告

AI Coding 六个月真实ROI账本:产品经理的血泪教训,研发的冷静忠告

6个月前的2025年12月,Boris Cherny 公开宣布自己卸载了 IDE。一时间,Vibe Coding 成了全行业最热的话题。6个月后,当我们回过头来拉一份真实账本,发现事情远没有"一句话生成一个App"那么浪漫。本文从产品经理和研发两个…

2026/6/30 4:03:30阅读更多 →
审计来了,数据权限全开——审计走了,怎么确保权限全部关掉?

审计来了,数据权限全开——审计走了,怎么确保权限全部关掉?

引言:审计结束三个月了,审计员的权限还没关某城商行每年按照监管要求开展至少一次数据安全审计。审计期间,内审部门需要抽样检查各类业务数据——交易流水、客户信息、员工操作日志、权限配置记录。这些数据分布在不同系统中,审计…

2026/6/30 4:36:27阅读更多 →
YOLOv8推理性能优化:从1.2FPS到35FPS的全链路加速实践

YOLOv8推理性能优化:从1.2FPS到35FPS的全链路加速实践

如果你在部署 YOLOv8 时,发现推理速度只有可怜的 1-2 FPS,而别人的演示视频却能跑到 30 FPS 以上,那么问题很可能不在模型本身,而在于你的整个处理链路。很多开发者拿到一个训练好的 YOLOv8 模型后,会直接使用官方示例…

2026/7/1 0:01:44阅读更多 →
Coze与Dify对比指南:低代码AI应用开发从入门到实战

Coze与Dify对比指南:低代码AI应用开发从入门到实战

1. 从零到一:为什么你需要了解 Coze 和 Dify?如果你对 AI 应用开发感兴趣,但一看到“大模型”、“智能体”、“工作流”这些词就头疼,觉得门槛太高,那这篇文章就是为你准备的。很多开发者,包括我自己&#…

2026/7/1 0:01:44阅读更多 →
AI生图工具怎么选?2026年6月版实测对比

AI生图工具怎么选?2026年6月版实测对比

做自媒体的朋友应该都有体会:配图一直是个让人头疼的问题。2026年,AI生图工具已经非常成熟了,但工具太多反而不知道怎么选。以下是截至2026年6月我对主流AI生图工具的实测对比。Midjourney V8.1:速度之王2026年6月11日&#xff0c…

2026/7/1 0:01:44阅读更多 →
YOLOv8推理性能优化:从1.2FPS到35FPS的全链路加速实践

YOLOv8推理性能优化:从1.2FPS到35FPS的全链路加速实践

如果你在部署 YOLOv8 时,发现推理速度只有可怜的 1-2 FPS,而别人的演示视频却能跑到 30 FPS 以上,那么问题很可能不在模型本身,而在于你的整个处理链路。很多开发者拿到一个训练好的 YOLOv8 模型后,会直接使用官方示例…

2026/7/1 0:01:44阅读更多 →
Coze与Dify对比指南:低代码AI应用开发从入门到实战

Coze与Dify对比指南:低代码AI应用开发从入门到实战

1. 从零到一:为什么你需要了解 Coze 和 Dify?如果你对 AI 应用开发感兴趣,但一看到“大模型”、“智能体”、“工作流”这些词就头疼,觉得门槛太高,那这篇文章就是为你准备的。很多开发者,包括我自己&#…

2026/7/1 0:01:44阅读更多 →
AI生图工具怎么选?2026年6月版实测对比

AI生图工具怎么选?2026年6月版实测对比

做自媒体的朋友应该都有体会:配图一直是个让人头疼的问题。2026年,AI生图工具已经非常成熟了,但工具太多反而不知道怎么选。以下是截至2026年6月我对主流AI生图工具的实测对比。Midjourney V8.1:速度之王2026年6月11日&#xff0c…

2026/7/1 0:01:44阅读更多 →