形态学操作

学习目标

  • 理解图像的邻域,连通性

  • 了解不同的形态学操作:腐蚀,膨胀,开闭运算,礼帽和黑帽等,及其不同操作之间的关系

1 连通性

在图像中,最小的单位是像素,每个像素周围有8个邻接像素,常见的邻接关系有3种:4邻接、8邻接和D邻接。分别如下图所示:
在这里插入图片描述

  • 4邻接:像素p(x,y)的4邻域是:(x+1,y);(x-1,y);(x,y+1);(x,y-1),用 N 4 ( p ) N_4(p) N4(p)表示像素p的4邻接

  • D邻接:像素p(x,y)的D邻域是:对角上的点 (x+1,y+1);(x+1,y-1);(x-1,y+1);(x-1,y-1),用 N D ( p ) N_D(p) ND(p)表示像素p的D邻域

  • 8邻接:像素p(x,y)的8邻域是: 4邻域的点 + D邻域的点,用 N 8 ( p ) N_{8}(p) N8(p)表示像素p的8邻域

连通性是描述区域和边界的重要概念,两个像素连通的两个必要条件是:

  • 两个像素的位置是否相邻

  • 两个像素的灰度值是否满足特定的相 似性准则(或者是否相等)
    根据连通性的定义,有4联通、8联通和m联通三种。

根据连通性的定义,有4联通、8联通和m联通三种。

  • 4联通:对于具有值V的像素p和q,如果q在集合 N 4 ( p ) N_4(p) N4(p)中,则称这两个像素是4连通。

  • 8联通:对于具有值V的像素p和q,如果q在集 合 N 8 ( p ) N_8(p) N8(p)中,则称这两个像素是8连通。

在这里插入图片描述

  • 对于具有值V的像素p和q,如果:
  1. q在集合 N 4 ( p ) N_4(p) N4(p)

  2. 或q在集合 N D ( p ) N_D(p) ND(p)中,并且 N 4 ( p ) N_4(p) N4(p) N 4 ( q ) N_4(q) N4(q)的交集为空(没有值V的像素)

则称这两个像素是m连通的,即4连通和D连通的混合连通。

在这里插入图片描述

2 形态学操作

形态学转换是基于图像形状的一些简单操作。它通常在二进制图像上执行。腐蚀和膨胀是两个基本的形态学运算符。然后它的变体形式如开运算,闭运算,礼帽黑帽等。

2.1 腐蚀和膨胀

腐蚀和膨胀是最基本的形态学操作,腐蚀和膨胀都是针对==白色部分(高亮部分)==而言的。

膨胀就是使图像中高亮部分扩张,效果图拥有比原图更大的高亮区域腐蚀是原图中的高亮区域被蚕食,效果图拥有比原图更小的高亮区域。膨胀是求局部最大值的操作,腐蚀是求局部最小值的操作。

  1. 腐蚀
    腐蚀的方法是,拿B的中心点和结构A上的点一个一个地对比,如果B上的所有点都在A的范围内,则该点保留,否则将该点去掉;右边是腐蚀后的结果。可以看出,它仍在原来A的范围内,且比A包含的点要少,就像A被腐蚀掉了一层。
    参考资料:图像处理:图像腐蚀、膨胀,开操作、闭操作
    在这里插入图片描述
    腐蚀的作用是消除物体边界点,使目标缩小,可以消除小于结构元素的噪声点。
    API:
   cv.erode(img,kernel,iterations)

参数:

  • img: 要处理的图像
  • kernel: 核结构
  • iterations: 腐蚀的次数,默认是1
  1. 膨胀
    具体操作是:用一个结构元素扫描图像中的每一个像素,用结构元素中的每一个像素与其覆盖的像素做“与”操作,如果都为0,则该像素为0,否则为1。如下图所示,结构A被结构B膨胀后:

在这里插入图片描述
膨胀的作用是将与物体接触的所有背景点合并到物体中,使目标增大,可添补目标中的孔洞。

API:

cv.dilate(img,kernel,iterations)

参数:

  • img: 要处理的图像

  • kernel: 核结构

  • iterations: 腐蚀的次数,默认是1

示例
我们使用一个5*5的卷积核实现腐蚀和膨胀的运算:

import numpy as np
import cv2 as cv
import matplotlib.pyplot as plt
# 1 读取图像
img = cv.imread("F:/images/7.png")
# 2 创建核结构
kernel = np.ones((5, 5), np.uint8)

# 3 图像腐蚀和膨胀
erosion = cv.erode(img, kernel) # 腐蚀
dilate = cv.dilate(img,kernel) # 膨胀

# 4 图像展示

# 解决中文显示问题
plt.rcParams['font.sans-serif'] = ['SimHei']
plt.rcParams['axes.unicode_minus'] = False

fig,axes=plt.subplots(nrows=1,ncols=3,figsize=(10,8),dpi=100)
axes[0].imshow(img)
axes[0].set_title("原图jsajdfj")
axes[1].imshow(erosion)
axes[1].set_title("腐蚀后结果")
axes[2].imshow(dilate)
axes[2].set_title("膨胀后结果")
plt.show()

在这里插入图片描述

2.2 开闭运算

开运算和闭运算是将腐蚀和膨胀按照一定的次序进行处理。 但这两者并不是可逆的,即先开后闭并不能得到原来的图像。

  1. 开运算
    开运算是先腐蚀后膨胀
  • 作用是:分离物体,消除小区域。
  • 特点:消除噪点,去除小的干扰块,而不影响原来的图像。
    在这里插入图片描述
    2. 闭运算

闭运算与开运算相反,是先膨胀后腐蚀,作用是消除/“闭合”物体里面的孔洞,特点:可以填充闭合区域。
在这里插入图片描述
3. API

cv.morphologyEx(img, op, kernel)

参数:

  • img: 要处理的图像
  • op: 处理方式:若进行开运算,则设为cv.MORPH_OPEN,若进行闭运算,则设为cv.MORPH_CLOSE
  • Kernel: 核结构
  1. 示例
    使用10*10的核结构对卷积进行开闭运算的实现。
import numpy as np
import cv2 as cv
import matplotlib.pyplot as plt
# 1 读取图像
img1 = cv.imread("./image/image5.png")
img2 = cv.imread("./image/image6.png")
# 2 创建核结构
kernel = np.ones((10, 10), np.uint8)
# 3 图像的开闭运算
cvOpen = cv.morphologyEx(img1,cv.MORPH_OPEN,kernel) # 开运算
cvClose = cv.morphologyEx(img2,cv.MORPH_CLOSE,kernel)# 闭运算
# 4 图像展示
fig,axes=plt.subplots(nrows=2,ncols=2,figsize=(10,8))
axes[0,0].imshow(img1)
axes[0,0].set_title("原图")
axes[0,1].imshow(cvOpen)
axes[0,1].set_title("开运算结果")
axes[1,0].imshow(img2)
axes[1,0].set_title("原图")
axes[1,1].imshow(cvClose)
axes[1,1].set_title("闭运算结果")
plt.show()

在这里插入图片描述

2.3 礼帽和黑帽

  1. 礼帽运算

原图像与“开运算“的结果图之差,如下式计算:

在这里插入图片描述
因为开运算带来的结果是放大了裂缝或者局部低亮度的区域,因此,从原图中减去开运算后的图,得到的效果图突出了比原图轮廓周围的区域更明亮的区域,且这一操作和选择的核的大小相关

礼帽运算用来分离比邻近点亮一些的斑块。当一幅图像具有大幅的背景的时候,而微小物品比较有规律的情况下,可以使用顶帽运算进行背景提取

  1. 黑帽运算

为”闭运算“的结果图与原图像之差。数学表达式为:
在这里插入图片描述
黑帽运算后的效果图突出了比原图轮廓周围的区域更暗的区域,且这一操作和选择的核的大小相关。

黑帽运算用来分离比邻近点暗一些的斑块。

  1. API
cv.morphologyEx(img, op, kernel)

参数:

  • img: 要处理的图像

  • op: 处理方式:

在这里插入图片描述

  • Kernel: 核结构
import numpy as np
import cv2 as cv
import matplotlib.pyplot as plt
# 1 读取图像
img1 = cv.imread("./image/image5.png")
img2 = cv.imread("./image/image6.png")
# 2 创建核结构
kernel = np.ones((10, 10), np.uint8)
# 3 图像的礼帽和黑帽运算
cvOpen = cv.morphologyEx(img1,cv.MORPH_TOPHAT,kernel) # 礼帽运算
cvClose = cv.morphologyEx(img2,cv.MORPH_BLACKHAT,kernel)# 黑帽运算
# 4 图像显示
fig,axes=plt.subplots(nrows=2,ncols=2,figsize=(10,8))
axes[0,0].imshow(img1)
axes[0,0].set_title("原图")
axes[0,1].imshow(cvOpen)
axes[0,1].set_title("礼帽运算结果")
axes[1,0].imshow(img2)
axes[1,0].set_title("原图")
axes[1,1].imshow(cvClose)
axes[1,1].set_title("黑帽运算结果")
plt.show()

在这里插入图片描述

  • 2
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Matlab中的形态学操作主要有膨胀、腐蚀、开运算、闭运算、骨架提取等。 1. 膨胀操作: 膨胀操作是一种图像处理中常用的形态学操作,它可以将图像中的物体进行膨胀,使物体的边界向外扩张。膨胀操作可以用于填充图像中的空洞、连接相邻物体等。 使用Matlab中的imdilate函数进行膨胀操作,例如: ```matlab img = imread('lena.jpg'); se = strel('disk', 5); img_dilate = imdilate(img, se); imshow(img_dilate); ``` 其中,img是待处理的图像,se是膨胀操作的结构元素,可以是各种形状,例如矩形、圆形等。上述代码中,我们使用了一个半径为5的圆形结构元素进行膨胀操作。 2. 腐蚀操作: 腐蚀操作是一种图像处理中常用的形态学操作,它可以将图像中的物体进行腐蚀,使物体的边界向内收缩。腐蚀操作可以用于去除图像中的噪声、分离相邻物体等。 使用Matlab中的imerode函数进行腐蚀操作,例如: ```matlab img = imread('lena.jpg'); se = strel('disk', 5); img_erode = imerode(img, se); imshow(img_erode); ``` 其中,img是待处理的图像,se是腐蚀操作的结构元素,可以是各种形状,例如矩形、圆形等。上述代码中,我们使用了一个半径为5的圆形结构元素进行腐蚀操作。 3. 开运算操作: 开运算操作是一种图像处理中常用的形态学操作,它可以先进行腐蚀操作,再进行膨胀操作,用于去除图像中的噪声、平滑物体的边缘等。 使用Matlab中的imopen函数进行开运算操作,例如: ```matlab img = imread('lena.jpg'); se = strel('disk', 5); img_open = imopen(img, se); imshow(img_open); ``` 其中,img是待处理的图像,se是开运算操作的结构元素,可以是各种形状,例如矩形、圆形等。上述代码中,我们使用了一个半径为5的圆形结构元素进行开运算操作。 4. 闭运算操作: 闭运算操作是一种图像处理中常用的形态学操作,它可以先进行膨胀操作,再进行腐蚀操作,用于填充图像中的空洞、平滑物体的边缘等。 使用Matlab中的imclose函数进行闭运算操作,例如: ```matlab img = imread('lena.jpg'); se = strel('disk', 5); img_close = imclose(img, se); imshow(img_close); ``` 其中,img是待处理的图像,se是闭运算操作的结构元素,可以是各种形状,例如矩形、圆形等。上述代码中,我们使用了一个半径为5的圆形结构元素进行闭运算操作。 5. 骨架提取: 骨架提取是一种图像处理中常用的形态学操作,它可以将图像中的物体提取出来,得到物体的骨架结构。骨架提取可以用于医学图像分析、机器人视觉等领域。 使用Matlab中的bwmorph函数进行骨架提取操作,例如: ```matlab img = imread('lena.jpg'); img_bw = im2bw(img); img_skel = bwmorph(img_bw, 'skel', Inf); imshow(img_skel); ``` 其中,img是待处理的图像,先将图像转换为二值图像img_bw,然后使用bwmorph函数进行骨架提取操作,‘skel’表示提取骨架结构,Inf表示提取到最细的骨架结构。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值