二值形态学——腐蚀与膨胀 及 C语言代码实现
参考文献:数字图像处理(第三版) 何东健 西安电子科技大学出版社
二值形态学中的运算对象是集合, 但实际运算中, 当涉及两个集合时并不把它们看作是互相对等的。 一般设A为图像集合, S为结构元素, 数学形态学运算是用S对A进行操作。 结构元素本身也是一个图像集合, 不过通常其尺寸要比目标图像小得多。 对结构元素可指定一个原点, 将其作为结构元素参与形态学运算的参考点。 原点可包含在结构元素中, 也可不包含在结构元素中, 但运算的结果常不相同。 以下用黑点代表值为1的区域, 白点代表值为0的区域, 运算对于值为1的区域进行。
1.腐蚀
腐蚀是一种最基本的数学形态学运算。 对给定的目标图像X和结构元素S, 将S在图像上移动, 则在每一个当前位置x, S+x只有3种可能的状态, 如下图所示:

第(1)种情形说明S+x与X相关;
第(2)种情形说明S+x与X不相关;
第(3)种情形说明S+x与X只是部分相关。
因而满足(1)式的点x的全体元素,称该点集为S对X的腐蚀(简称腐蚀, 也称X用S腐蚀),记为
。
腐蚀也可以用集合的方式定义:
该式表明, X用S腐蚀的结果是所有使S平移x后仍在X中的x的集合。 换句话说, 用S来腐蚀X得到的集合是S完全包含在X中时S的原点位置的集合。
腐蚀在数学形态学运算中的作用是消除物体边界点、 去除小于结构元素的物体、 清除两个物体间的细小连通等。 如果结构元素取3×3的像素块, 腐蚀将使物体的边界沿周边减少1个像素。
“腐蚀”图解:(腐蚀将图像(区域)缩小了)


代码实现:
【注】二值腐蚀基本运算,背景为黑色,目标为白色。
//二值腐蚀
/*函数参数:
a——待腐蚀的图像
b——腐蚀后的结果
mat[5][5]——结构元素,我这里默认设了5*5的大小
*/
void Bi_Corrosion(Mat &a, Mat &b, int mat[][])
{
int i, j, k, o;
int rows = a.rows;
int cols = a.cols*a.channels(); bool flag; uchar *dst = b.data;
uchar *src = a.data;
//针对图像中每一个像素位置,判断是否结构元素能填入目标内部
for(i = ; i < rows-; i++) {
for(j = ; j < cols-; j++) {
//判断结构元素是否可以在当前点填入目标内部,1为可以,0为不可以
flag = ;
for(k = -; k <= && flag; k++) {
for(o = -; o <= ; o++) {
//如果当前结构元素位置为1,判断与对应图像上的像素点是否为非0
if(mat[k+][o+]) {
//如果图像当前像素为0,则没有击中该点,不是腐蚀的输出
if(!*(src+(i+k)*cols+j+o)){
flag = ; break;
}
}
}
}
*(dst+i*cols+j) = flag ? : ;
}
}
}
2.膨胀
腐蚀可以看作是将图像X中每一个与结构元素S全等的子集S+x收缩为点x。 反之, 也可以将X中的每一个点x扩大为S+x, 即膨胀运算, 记为
。用集合语言定义膨胀运算的定义形式为:
图示:

【注意】来看下特殊情况: 用B膨胀后,结果向左平移了;而用B图像的反射膨胀后位置不变。

对于非对称结构S,膨胀后会使得原图错移,但
膨胀不会,总的位置和形状不变,因此膨胀公式也可以写做:

对集合X的膨胀也可以看做是对集合X补集的腐蚀的补集具有对偶特性:

腐蚀和膨胀运算与集合运算的关系如下:


代码实现:
【注】二值膨胀基本运算,背景为黑色,目标为白色。
//二值膨胀
/*函数参数:
a——待腐蚀的图像
b——腐蚀后的结果
mat——结构元素
*/
void Bi_Expansion(Mat &a, Mat &b, int mat[][]) {
int i, j, k, o;
int rows = a.rows;
int cols = a.cols*a.channels();
Mat tmp = a.clone();
uchar* src = tmp.data;
//膨胀是对图像中目标补集的腐蚀,因此先求输入图像数据的补集
for(i = ; i < rows; i++)
for(j = ; j < cols; j++)
*(src+i*cols+j) = - *(src+i*cols+j);
//膨胀是结构元素的对称集对补集的腐蚀,此处求其反射
for(i = ; i < ; i++)
for(j = ; j <= i; j++)
mat[i][j] = mat[j][i];
bool flag;
uchar* dst = b.data;
//针对图像中每一个像素位置,判断是否结构元素能填入目标内部
for(i = ; i < rows-; i++) {
for(j = ; j < cols-; j++) {
//判断结构元素是否可以在当前点填入目标内部,1为可以,0为不可以
flag = ;
for(k = -; k <= && flag; k++) {
for(o = -; o <= ; o++) {
//如果当前结构元素位置为1,判断与对应图像上的像素点是否为非0
if(mat[k+][o+]) {
if(!*(src+(i+k)*cols+j+o)){//没击中
flag = ; break;
}
}
}
}
*(dst+i*cols+j) = flag ? : ;
}
}
//用结构元素对称集对目标补集腐蚀后,还要对结构再求一次补集,才是膨胀结构输出
//赋值结构元素腐蚀漏掉的区域,使原图像恢复为二值图像
for(i = ; i < rows; i++) {
for(j = ; j < cols; j++) {
*(dst+i*cols+j) = - *(dst+i*cols+j);
if(*(dst+i*cols+j) != && *(dst+i*cols+j) != )
*(dst+i*cols+j) = ;
}
}
}
二值形态学——腐蚀与膨胀 及 C语言代码实现的更多相关文章
- OpenCV学习 7:图像形态学:腐蚀、膨胀
原创文章,欢迎转载,转载请注明出处 首先什么是图像形态学?额,这个抄下百度到的答案.基本思想: 用具有一定形态的结构元素去度量和提取图像中的对应形状已达到对图像分析和识别的目的,形态学图像处理表 ...
- opencv中的图像形态学——腐蚀膨胀
腐蚀膨胀是图像形态学比较常见的处理,腐蚀一般可以用来消除噪点,分割出独立的图像元素等. 一般腐蚀操作对二值图进行处理,腐蚀操作如下图,中心位置的像素点是否与周围领域的像素点颜色一样(即是否是白色点,即 ...
- OpenCV学习笔记(六) 滤波器 形态学操作(腐蚀、膨胀等)
转自:OpenCV 教程 另附:计算机视觉:算法与应用(2012),Learning OpenCV(2009) 平滑图像:滤波器 平滑 也称 模糊, 是一项简单且使用频率很高的图像处理方法.平滑处理的 ...
- 机器学习进阶-项目实战-信用卡数字识别 1.cv2.findContour(找出轮廓) 2.cv2.boudingRect(轮廓外接矩阵位置) 3.cv2.threshold(图片二值化操作) 4.cv2.MORPH_TOPHAT(礼帽运算突出线条) 5.cv2.MORPH_CLOSE(闭运算图片内部膨胀) 6. cv2.resize(改变图像大小) 7.cv2.putText(在图片上放上文本)
7. cv2.putText(img, text, loc, text_font, font_scale, color, linestick) # 参数说明:img表示输入图片,text表示需要填写的 ...
- opencv 4 图像处理(2 形态学滤波:腐蚀与膨胀,开运算、闭运算、形态学梯度、顶帽、黑帽)
腐蚀与膨胀 膨胀(求局部最大值)(dilate函数) #include <opencv2/core/core.hpp> #include <opencv2/highgui/highg ...
- c#数字图像处理(十二)图像的腐蚀与膨胀
背景知识 腐蚀与膨胀基本原理:就是用一个特定的结构元素来与待处理图像按像素做逻辑操作:可以理解成拿一个带孔的网格板(结构元素矩阵中元素为1的为孔)盖住图像的某一部分,然后按照各种不同的观察方式来确定操 ...
- 【python-opencv】17-形态学操作-腐蚀与膨胀
形态学操作其实就是改变物体的形状,比如腐蚀就是"变瘦",膨胀就是"变胖",看下图就明白了: 形态学操作一般作用于二值化图(也可直接作用于原图),来连接相邻的元素 ...
- OpenCV腐蚀与膨胀(Eroding and Dilating)
腐蚀与膨胀(Eroding and Dilating) 目标 本文档尝试解答如下问题: 如何使用OpenCV提供的两种最基本的形态学操作,腐蚀与膨胀( Erosion 与 Dilation): ero ...
- ComicEnhancerPro 系列教程十七:二值化图像去毛刺
作者:马健邮箱:stronghorse_mj@hotmail.com 主页:http://www.comicer.com/stronghorse/ 发布:2017.07.23 教程十七:二值化图像去毛 ...
随机推荐
- iOS开源项目周报0223
由OpenDigg 出品的iOS开源项目周报第九期来啦.我们的iOS开源周报集合了OpenDigg一周来新收录的优质的iOS开源项目,方便iOS开发人员便捷的找到自己需要的项目工具等.panelkit ...
- Jquery Ajax 复杂json对象提交到WebService
在ajax的已不请求中,常常返回json对象.可以利用json.net给我们提供的api达到快速开发. 例子: using System;using System.Collections;using ...
- DataGridView 隔行显示不同的颜色
两种方法 第一种 DataGridview1.Rows[i].DefultCellStyle.backcolor 第二种 AlternatingRowsDefutCellstyle 属性 获取或设置应 ...
- WAMP配置httpd.conf允许外部访问
在电脑上开启Apache服务后,如何让外部网络访问呢? 在网上查找答案和问过一些小伙伴后,得到以以下方案.大致是在httpd.conf中加入一些语句以及利用自己的WiFi建立热点,让需要访问的设备连接 ...
- [C语言] 数据结构-预备知识结构体
结构体 为什么出现结构体 为了表示一些复杂的数据,而普通的基本类型变量无法满足需求 什么叫结构体 结构体是用户根据实际需要自己定义的复合数据类型 如何使用结构体 1.两种方式 一般使用结构体指针的形 ...
- 在jsp里调用out.flush()和response.flushBuffer()有什么区别
out.flush(); out是PrintWriter的实例 public void flush() Flush the stream. response.flushBuffer(): ...
- 04-Tomcat体系结构与插件配置
一.发布程序详解 Context docBase:web应用的文件路径 path:URL入口 reloadable:字节码变化服务器是否重新加载web应用 二.tomcat服务器体系结构 1.Serv ...
- MyBatis学习(二)---数据表之间关联
想要了解MyBatis基础的朋友可以通过传送门: MyBatis学习(一)---配置文件,Mapper接口和动态SQL http://www.cnblogs.com/ghq120/p/8322302. ...
- 《JavaWeb从入门到改行》注册时向指定邮箱发送邮件激活
javaMail API javaMail是SUN公司提供的针对邮件的API . 两个jar包 mail.jar 和 activation.jar java mail中主要类:javax.mail. ...
- PostgreSQL Metadata
http://www.devart.com/dotconnect/postgresql/docs/MetaData.html In this overload first parameter is ...