Bootstrap

【OpenCV】漫水填充

1、漫水填充的定义

漫水填充法是一种特定的颜色填充连通区域,通过设置可以连通像素的上下限以及连通方式来达到不同的填充效果的方法。漫水填充经常被用来标记或者分离图像的一部分,以便对其进行进一步处理或分析,也可以用来从输入图像获取掩码区域,掩码会加速处理过程,或只处理掩码指定的像素点,操作的结果总是某个连续的区域。

补充:图像掩码相关知识:

1、图象掩码这个很多种类;二值掩码就是设定一个阈值,如果像素的灰度大于该阈值这变为黑,小于则变为白,图像经过此变换就变成二值图像。当然如果是彩色图像可以先转化为灰度图。

2、二值图像是每个像素只有两个可能值的数字图像。人们经常用黑白、B&W、单色图像表示二值图像,但是也可以用来表示每个像素只有一个采样值的任何图像,例如灰度图像等。二值图像经常出现在数字图像处理中作为图像掩码或者在图像分割、二值化和dithering的结果中出现。一些输入输出设备,如激光打印机、传真机、单色计算机显示器等都可以处理二值图像。

2、漫水填充法的基本思想

所谓漫水填充,就是自动选中了和种子点相连的区域,接着将该点区域替换成指定的颜色,经常用来标记或者分离图像的一部分进行处理或分析。漫水填充也可以用来从输入图像获取掩码区域,掩码会加速处理过程,或者只处理掩码指定的像素点。

以此填充算法为基础,类似PhotoShop的魔术棒选择工具就很容易实现了。漫水填充(FloodFill)是查找和种子点连通的颜色相同的点,魔术棒选择工具则是查找和种子点连通的颜色相近的点,把和初始种子像素颜色相近的点压进栈做为新种子。

3、实现漫水填充算法:floodFill函数

说明:漫水填充算法由floodFill函数实现,其作用是用我们指定的颜色从种子点开始填充一个连接域。连通性由像素值的接近程度来衡量。

两个版本的floodFill,一个不带掩膜mask版和带mask版本。这个掩膜mask,就是用于进一步控制哪些区域将被填充颜色(比如对于同一图像进行多次填充时)。这两个版本的FloodFill,都必须在图像中选一个种子点,然后把临近区域所有相似点填充上同样的颜色,不同的是,不一定将所有的邻近点都染上同一颜色,漫水填充操作的结果总是某个连接的区域,当邻近像素点位于给定的范围(从loDiff到upDiff)内或在原始seedPoint像素值范围内时,FloodFill函数就会为这个点涂上颜色。具体如下:

  • 版本一
int floodFill(InputOutputArray image,Point seedPoint,Scalar newVal,Rect * rect=0,Scalar loDiff=Scalar(),Scalar upDiff=Scalar(),int flags=4);
  • 版本二
int floodFill(InputOutputArray image,IntputOutputArray mask,Point seedPoint,Scalar newVal,Rect * rect=0,Scalar loDiff=Scalar(),Scalar upDiff=Scalar(),int flags=4)
  • 第一个参数:输入/输出1通道或3通道,8位或浮点图像,具体参数由之后的参数指明。

  • 第二个参数:第二个版本独享的参数,表示操作掩膜。它应该为单通道,8位,长和宽上都比输入图像image大两个像素点的图像。第二个版本的floodFill需要使用以及更新掩膜,所以对于这个mask参数,我们一定要将其准备好并填在此处。需要注意的是,漫水填充不会填充掩膜mask的非零像素区域。如:一个边缘检测算子的输出可以用来作为掩膜,以防止填充到边缘。同样的,也可以在多次的函数调用中使用同一个掩膜,以保证填充的区域不会重叠。另需要注意的是,掩膜mask会比需填充的图像大,所以mask中与输入图像(x,y)像素点相对应的点的坐标为(x+1,y+1)

  • 第三个参数:Point seed类型的Point漫水填充的起始点

  • 第四个参数:Scalar 类型的newVal,像素点被染色的值,即在重绘区域像素的新值

  • 第五个参数:Rect *类型的rect,有默认值0,一个可选的参数,用于设置floodFill函数将要重绘区域的最小边界矩形区域

  • 第六个参数:Scalar 类型的loDiff,有默认值Scalar(),表示当前观察像素值与其部件邻域像素值或者待加入该部件的种子像素之间的亮度或颜色之负差(lower brightness/color difference)的最大值

  • 第七个参数:Scalar 类型upDiff,有默认值Scalar(),表示当前观察像素值与其部件邻域像素值或者待加入该部件的种子像素之间的亮度或颜色之正差(lower brightness/color difference)的最大值

  • 第八个参数:int 类型的flags,操作标识符,此参数包含三个部分

    (1)低八位(0~7位)用于控制算法的连通性,可取4(4为默认值)或者8.如果设为4,表示填充算法只考虑当前像素水平方向和垂直方向的相邻点;如果设为8,除上述相邻点之外,还会包含对角线方向的相邻点

    (2)高八位部分(16~32位)可以为0或者如下两种选项标识符的组合

    • FLOODFILL_FIXED_RANGE:如果设置这个标识符,就会考虑当前像素与种子像素之间的差,否则就考虑当前像素与其像素与相邻像素的差。这个范围是浮动的
    • FLOODFILL_MASK_ONLY:如果设置这个标识符,函数不会去填充改变原始图像(也就忽略第三个参数new Val),而是去填充掩膜图像(mask)。这个标识符只对第二个版本的floodFill有用。

    (3)中间八位部分(8~15),上面高八位FLOODFILL_MASK_ONLY标识符已经说的很明显,需要输入符合要求的掩码。floodFill的flags参数的中间八位的值就是用于指定填充掩码图像的值,如果flags中间八位的值为0,则掩码会用1来填充

    所有flags可以用or操作符连接起来,即“ | ”。如:用8邻域填充,并填充固定像素值范围,填充掩码而不是填充源图像,以及设置填充值为38,那么输入参数如下这样:

    flags=8|FLOODFILL_MASK_ONLY|FLOODFILL_FIXED_RANGE|(38<<8)
    

4、代码详解

#include<opencv2/opencv.hpp>
#include<opencv2/imgproc/imgproc.hpp>
using namespace cv;
int main()
{
	Mat src = imread("E:\\Pec\\蜘蛛侠.jpg");
	imshow("【原始图】", src);
	Rect ccomp;
	floodFill(src, Point(100, 500), Scalar(155, 255, 55), &ccomp, Scalar(20, 20, 20),
	 Scalar(20, 20, 20));
	imshow("【效果图】", src);
	waitKey(0);
}

在这里插入图片描述
在这里插入图片描述

;