(OpenCV+Python)图像腐蚀膨胀 开闭运算 线性滤波 非线性滤波.docx
图像腐蚀与膨胀我们在前两次教程中概述了 OPenCV对于图像的滤波,通常对于一个实战项 目而言,滤波之后的下一步操作就是图像的形态学处理了,从本次教程开始,我 们正式步入了 OPenCV图像形态学处理的部分。形态学(morphology) 一词通常表示生物学的一个分支,该分支主要研究动 植物的形态和结构。而我们图像处理中指的形态学,往往表示的是数学形态学。 下面一起来了解数学形态学的概念。数学形态学是一门建立在格论和拓扑学基础之上的图像分析学科,是数学形 态学图像处理的基本理论。其基本的运算包括:二值腐蚀和膨胀、二值开闭运算、 骨架抽取、极限腐蚀、击中击不中变换、形态学梯度、ToP-hat变换、颗粒分析、 流域变换、灰值腐蚀和膨胀、灰值开闭运算、灰值形态学梯度等。简单来讲,形态学操作就是基于形状的一系列图像处理操作。OPenCV为进 行图像的形态学变换提供了快捷、方便的函数。最基本的形态学操作有二种,他 们是:膨胀与腐蚀(DilatiOn与Erosion)o膨胀与腐蚀能实现多种多样的功能,主要如下消除噪声。分割出独立的图像元素,在图像中连接相邻的元素。寻找图像中的明显的极大值区域或极小值区域。求出图像的梯度。腐蚀和膨胀是对白色部分(高亮部分)而言的,不是黑色部分。膨胀就是图 像中的高亮部分进行膨胀,“领域扩张”,效果图拥有比原图更大的高亮区域。腐 蚀就是原图中的高亮部分被腐蚀,”领域被蚕食”,效果图拥有比原图更小的高亮 区域。月彭胀其实:膨胀就是求局部最大值的操作。按数学方面来说,膨胀或者腐蚀操作就是将图像(或图像的一部分区域,我 们称之为A)与核(我们称之为B)进行卷积。核可以是任何的形状和大小,它拥有一个单独定义出来的参考点,我们称其 为锚点。多数情况下,核是一个小的中间带有参考点和实心正方形或者圆盘,其 实,我们可以把核视为模板或者掩码。而膨胀就是求局部最大值的操作,核B与图形卷积,即计算核B覆盖的区 域的像素点的最大值,并把这个最大值赋值给参考点指定的像素。这样就会使图 像中的高亮区域逐渐增长。如下图所示,这就是膨胀操作:我们来看一下函数原型:cv2. dilate (img,kernel,iterations)->dst第一个参数:img指需要膨胀的图。第二个参数:kernel指膨胀操作的内核,默认是一个简单的3X3矩阵,我们 也可以利用getStructuringElement O函数指明它的形状。第三个参数:iterations指的是膨胀次数,省略是默认为1。dst则为返回的图像。定义卷积核需要用到Numpy中的函数,它可以定义一个矩形的卷积核结构 元素,我们来看一下代码:import cv2import numpy as npimg = cv2.imread(,01 .jpg,O)kernel =np.ones(5,5),np.uint8)dict = cv2.dilate(img,kernel,iterations =I)cv2.imshow(,org,img)cv2.imshow(resultn,dict)cv2.waitKey(0)cv2.destroy AllWindowsO我们定义了一个5*5的矩形卷积核,效果:HuX MulX事实上,在某些情况下,我们可能需要椭圆形/圆形的内核。因此,为此, OpenCV具有一个函数cv.getStructuringElement ()。我们只需传递内核的形状和 大小,即可获得所需的内核,函数原型:retval=cv.getStructuringElement(shape, ksize, anchor)这个函数的第一个参数表示内核的形状,有三种形状可以选择。矩形:MORPH_RECT交叉形:MORPH_CROSS椭圆形:MORPH_ELLIPSEIlll 1OOlO 00 0 10 01111111111001001111111111111111111111111001001111 100 10 000 10 0十字形结构矩形结构精国结构第二和第三个参数分别是内核的尺寸以及锚点的位置。一般在调用erode以 及dilate函数之前,先定义一个变量来获得。getStructuringElement函数的返U值:对于锚点的位置,有默认值Point(-1,-1),表示锚点位于中心点。element形状唯一依赖锚点位置,其他情况下, 锚点只是影响了形态学运算结果的偏移。我们看一下代码:import cv2import numpy as npimg = cv2.imread(,01 .jpg,O)kernel =cv2.getStructuringElement(cv2.MORPH_ELLIPSE,(5,5)dict =cv2.dilate(img,kernel,iterations = I)cv2.imshow(norgn,img)cv2.imshow(nresultn,dict)cv2.waitKey(0)cv2.destroy AllWindowsO至于卷积核的形状我在这里选择了椭圆形,大家可以自己选择其他形状进行 实验:腐蚀再来看一下腐蚀,膨胀和腐蚀是一对好友,是相反的一对操作,所以腐蚀就 是求局部最小值的操作。我们一般都会把腐蚀和膨胀对应起来理解和学习。下文 就可以看到,两者的函数原型也是基本上一样的。我们来看一下函数原型:cv2.erode(img,kernel,iterations)->dst第一个参数:img指需要腐蚀的图。第二个参数:kernel指腐蚀操作的内核,默认是一个简单的3X3矩阵,我们 也可以利用getStructuringElement O函数指明它的形状。第三个参数:iterations指的是腐蚀次数,省略是默认为1。代码:import cv2import numpy as npimg = cv2.imread(,01 .jpg,O)kernel =cv2.getStructuringElement(cv2.MORPH_ELLIPSE,(5,5)erode =cv2.erode(img,kernel,iterations = I)cv2.imshow(norgn,img)cv2.imshow(nresultn,erode)cv2.waitKey(0)cv2.destroy AllWindowsO效果:HuXZ JHll <腐蚀与膨胀是形态学处理中的基础操作,它们有着很重要的作用,也是后面 开操作与闭操作的基础,所以必须熟练运用。开运算与闭运算图像的腐蚀与膨胀是本次教程的核心一一开运算与闭运算的基础,如果结构 元素为圆形,则膨胀操作可填充图像中比结构元素小的孔洞以及图像边缘处小 的凹陷部分。而腐蚀可以消除图像中的毛刺及细小连接成分,并将图像缩小,从 而使其补集扩大。但是,膨胀和腐蚀并非互为逆运算,所以它们可以结合使用。 在腐蚀和膨胀两个基本运算的基础上,可以构造出形态学运算簇,它由膨胀和 腐蚀两个运算的复合与集合操作(并、交、补等)组合成的所有运算构成。例如, 可使用同一结构元素,先对图像进行腐蚀然后膨胀其结果,该运算称为开运算; 或先对图像进行膨胀然后腐蚀其结果,称其为闭运算。开运算和闭运算是形态 学运算族中两种最为重要的运算。对于图像X及结构元素S,用符号X OS表示S对图像X作开运算,用符号X S表示S对图像X作闭运算,它们的定义为:XoS= (X S) SX-S= (X S) S首先需要来了解一个函数:cv2.morphologyEx(src, op, kernel)src传入的图片。op进行变化的方式。kernel表示定义的卷积核的大小以及形状。op = cv2.MORPH_OPEN进行开运算,指的是先进行腐蚀操作,再进行膨 胀操作。op = cv2.MORPH_CLOSE进行闭运算,指的是先进行膨胀操作,再进行腐 蚀操作。开运算开运算指的就是对图像先进行腐蚀操作,然后再进行膨胀操作,而通常情况 下,它是对图像的明亮的区域进行操作,可以消除图像中的白噪声,现在我们来 看例子,先看一幅图像:现在我们想要消除图像中的黑色的毛刺,但是如果直接对图像进行开运算是 不行的,因为开运算是对图像的明亮区域进行操作,看一下直接进行开运算会有 什么效果:import cv2import numpy as npimg = cv2.imread(,open.jpg,0)kernel =c v2. getS tructuringElement(c v2. MORPH_RECT, (3,3) open=cv2.morphologyEx(img,cv2.MORPH-OPEN,krnel)cv2.imshow(nimg,img)cv2.imshow(result, open)cv2.waitKey(0)cv2.destroyAllWindowsO可以看到,图像的毛刺没有被去除,现在我们需要将原图进行阈值化翻转, 也就是黑白颠倒,这样才方便进行形态学的处理,我们在前面阈值部分讲过,这 里就不再讲述了,直接看代码:import cv2import numpy as npimg = cv2.imread(,open.jpg,0)threshold =cv2.threshold(img,0,255,cv2.THRESH_BINARY_INV|cv2.THRESH_OTSU)l cv2.imshow(,imgn,img)cv2.imshow(,thres,threshold)cv2.waitKey(0)cv2.destro y AllWindowsO现在图像已经被黑白颠倒了过来,现在我们可以开始进行开运算了,当然首 先也是需要定义一个卷积核的,这在上个教程中已经谈到,在这里我们定义一个 3*3的矩形卷积核:import cv2import numpy as npimg = cv2.imread(,open.jpg,0)threshold =cv2.threshold(img,0,255,cv2.THRESH_BINARYNVlCV2.THRESHJDTSU)1kernel = cv2.getStructuringElement(cv2.MORPH-RECT,(3,3)op=cv2.morphologyEx(threshold,cv2.MORPH-OPEN,kernel)cv2.imshow(img,img)cv2.imshow(thres,threshold)cv2.imshow(nresult,open)cv2.waitKey(0)cv2.destroy AllWindowsO这样效果就显而易见了,如果我们将卷积核改成5*5的呢:import cv2import numpy as npimg = cv2.imread(,open.jpg,0)threshold = cv2.threshold(img,0,255,cv2.THRESH_BINARYNVlCV2.THRESHJDTSU)1kernel = cv2.getStructuringElement(cv2.MORPH_RECT,(5,5)open=cv2.morphologyEx(threshold,cv2.MORPH-OPEN,kernel