OpenCV空洞填充算法




討論帖:

  在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






免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM