討論帖:
在Matlab下,使用imfill可以很容易的完成孔洞填充操作,感覺這是一個極為常用的方法,然而不知道為什么Opencv里面卻沒有集成這個函數。在網上查了好多關於Opencv下的孔洞填充方法,大部分使用輪廓查找方法去做的,但對於這種方法,總感覺不是特別好。之前了解過岡薩雷斯那本書上的孔洞填充算法,所以想着手重新寫一個。這里借鑒了岡薩雷斯書上的集合運算方法(並不完全一樣)
大致思路如下:
0, 設原圖像為 A。
1, 首先A向外延展一到兩個像素,並將值填充為背景色(0),標記為B。
2, 使用floodFill函數將B的大背景填充,填充值為前景色(255),種子點為(0,0)即可(步驟一可以確保(0,0)點位於大背景),標記為C。
3, 將填充好的圖像裁剪為原圖像大小(去掉延展區域),標記為D。
4, 將D取反與A相加即得填充的圖像,E=A|(~D)。
// fillhole.cpp : 定義控制台應用程序的入口點。
//
#include "stdafx.h"
#include <opencv2/core/core.hpp>
#include <opencv2/highgui/highgui.hpp>
#include "opencv2/imgproc/imgproc.hpp"
#pragma comment(lib,"opencv_core2410d.lib")
#pragma comment(lib,"opencv_highgui2410d.lib")
#pragma comment(lib,"opencv_imgproc2410d.lib")
using namespace std;
using namespace cv;
void fillHole(const Mat srcBw, Mat &dstBw)
{
Size m_Size = srcBw.size();
Mat Temp=Mat::zeros(m_Size.height+2,m_Size.width+2,srcBw.type());//延展圖像
srcBw.copyTo(Temp(Range(1, m_Size.height + 1), Range(1, m_Size.width + 1)));
cv::floodFill(Temp, Point(0, 0), Scalar(255));
Mat cutImg;//裁剪延展的圖像
Temp(Range(1, m_Size.height + 1), Range(1, m_Size.width + 1)).copyTo(cutImg);
dstBw = srcBw | (~cutImg);
}
int main()
{
Mat img=cv::imread("23.jpg");
Mat gray;
cv::cvtColor(img, gray, CV_RGB2GRAY);
Mat bw;
cv::threshold(gray, bw, 0, 255, CV_THRESH_BINARY | CV_THRESH_OTSU);
Mat bwFill;
fillHole(bw, bwFill);
imshow("填充前", gray);
imshow("填充后", bwFill);
waitKey();
return 0;
}
我調試好的工程:點擊打開鏈接
http://download.csdn.net/detail/wangyaninglm/9389329
