OpenCV提取圖像中圓線上的數(shù)據(jù)具體流程
需求說(shuō)明
在對(duì)圖像進(jìn)行處理時(shí),經(jīng)常會(huì)有這類(lèi)需求:客戶想要提取出圖像中某條直線、圓線或者ROI區(qū)域內(nèi)的感興趣數(shù)據(jù),進(jìn)行重點(diǎn)關(guān)注。該需求在圖像檢測(cè)領(lǐng)域尤其常見(jiàn)。ROI區(qū)域一般搭配Rect即可完成提取,直線和圓線數(shù)據(jù)的提取沒(méi)有現(xiàn)成的函數(shù),需要自行實(shí)現(xiàn)。
直線的提取見(jiàn):
OpenCV獲取圖像中直線上的數(shù)據(jù)具體流程
而圓線的提取則是本文要將的內(nèi)容,對(duì)圓線而言,將線上某點(diǎn)作為起點(diǎn),沿順時(shí)針或逆時(shí)針?lè)较蛞来翁崛「信d趣數(shù)據(jù),可放置在容器中。那么如何快速提取呢?本文提供了一種比較簡(jiǎn)單的思路,應(yīng)用窗口模板,在窗口中快速找到下一可前進(jìn)點(diǎn)的位置,步進(jìn)然后再找下個(gè)點(diǎn),形成路徑追蹤,進(jìn)而實(shí)現(xiàn)整圈圓線數(shù)據(jù)的提取。
具體流程
1)初始化。設(shè)置路徑追蹤窗口尺寸size為3,創(chuàng)建path作為行進(jìn)路徑,p點(diǎn)作為起點(diǎn),c用來(lái)存放目標(biāo)數(shù)據(jù)點(diǎn)。
cv::Mat c; int size = 3; cv::Mat path = mask.clone(); Point p = Point(center.x + radius, center.y);
2)將起點(diǎn)放置在c中,將path中的起點(diǎn)值置0,表示該點(diǎn)已經(jīng)走過(guò)。
c.push_back(src.at<uchar>(p.y, p.x)); path.at<uchar>(p.y, p.x) = 0;
3)用WinDataNum函數(shù)判斷當(dāng)前點(diǎn)的窗口內(nèi)有幾個(gè)可前進(jìn)點(diǎn),若無(wú)則說(shuō)明路徑封死或者完成路徑,wn值表示可前進(jìn)點(diǎn)的個(gè)數(shù)。
int wn = WinDataNum(path, p, size);
4)窗口內(nèi)遍歷,查看是否有可前進(jìn)路徑,若找到,則將當(dāng)前點(diǎn)信息刷新為此點(diǎn),并將標(biāo)記符find設(shè)為true,find的意義是快速中斷遍歷,用來(lái)提速。
int t = size / 2; bool find = false; for (int i = p.y - t; i <= p.y + t; ++i) { uchar *g = path.ptr<uchar>(i); for (int j = p.x - t; j <= p.x + t; ++j) { if (g[j] == 255) { p.x = j; p.y = i; find = true; break; } } if (find) break; }
5)若找到了點(diǎn),即find為true,則將該點(diǎn)的數(shù)據(jù)存放在c中,path中置0,并以該點(diǎn)為中心搜索窗口內(nèi)可前進(jìn)路徑。
if (find) { c.push_back(src.at<uchar>(p.y, p.x)); path.at<uchar>(p.y, p.x) = 0; wn = WinDataNum(path, p, size); } else break;
6)若wn為0了,則說(shuō)明路徑封死或者完成路徑了,跳出循環(huán),函數(shù)執(zhí)行完畢。?
while (wn) { int t = size / 2; bool find = false; for (int i = p.y - t; i <= p.y + t; ++i) { uchar *g = path.ptr<uchar>(i); for (int j = p.x - t; j <= p.x + t; ++j) { if (g[j] == 255) { p.x = j; p.y = i; find = true; break; } } if (find) break; } if (find) { c.push_back(src.at<uchar>(p.y, p.x)); path.at<uchar>(p.y, p.x) = 0; wn = WinDataNum(path, p, size); } else break; }
功能函數(shù)
// 獲取圓圈上的數(shù)據(jù),逆時(shí)針存儲(chǔ),起點(diǎn)在中心同行最右側(cè)數(shù)據(jù) cv::Mat getCircleData(cv::Mat src, cv::Mat mask, cv::Point center, int radius) { cv::Mat c; int size = 3; cv::Mat path = mask.clone(); Point p = Point(center.x + radius, center.y); c.push_back(src.at<uchar>(p.y, p.x)); path.at<uchar>(p.y, p.x) = 0; int wn = WinDataNum(path, p, size); while (wn) { int t = size / 2; bool find = false; for (int i = p.y - t; i <= p.y + t; ++i) { uchar *g = path.ptr<uchar>(i); for (int j = p.x - t; j <= p.x + t; ++j) { if (g[j] == 255) { p.x = j; p.y = i; find = true; break; } } if (find) break; } if (find) { c.push_back(src.at<uchar>(p.y, p.x)); path.at<uchar>(p.y, p.x) = 0; wn = WinDataNum(path, p, size); } else break; } return c; }
// 獲取窗口內(nèi)的有效數(shù)據(jù)個(gè)數(shù) int WinDataNum(cv::Mat path, cv::Point p, int size) { int number = 0; int t = size / 2; for (int i = p.y - t; i <= p.y + t; ++i) { uchar *g = path.ptr<uchar>(i); for (int j = p.x - t; j <= p.x + t; ++j) { if (g[j] == 255) number++; } } return number; }
C++測(cè)試代碼
#include <iostream> #include <opencv2/opencv.hpp> #include <string> using namespace std; using namespace cv; cv::Mat getCircleData(cv::Mat src, cv::Mat mask, cv::Point center, int radius); int WinDataNum(cv::Mat path, cv::Point p, int size); int main() { cv::Mat src = imread("test.jpg", 0); cv::Mat mask = cv::Mat::zeros(src.size(), CV_8UC1); cv::Point center = cv::Point(src.cols / 2, src.rows / 2); int radius = min(src.cols, src.rows) / 2 - 10; circle(mask, center, radius, Scalar(255), 1, 8); cv::Mat c = getCircleData(src, mask, center, radius); src.setTo(0, mask == 0); imshow("src", src); imshow("mask", mask); waitKey(0); return 0; } // 獲取圓圈上的數(shù)據(jù),逆時(shí)針存儲(chǔ),起點(diǎn)在中心同行最右側(cè)數(shù)據(jù) cv::Mat getCircleData(cv::Mat src, cv::Mat mask, cv::Point center, int radius) { cv::Mat c; int size = 3; cv::Mat path = mask.clone(); Point p = Point(center.x + radius, center.y); c.push_back(src.at<uchar>(p.y, p.x)); path.at<uchar>(p.y, p.x) = 0; int wn = WinDataNum(path, p, size); while (wn) { int t = size / 2; bool find = false; for (int i = p.y - t; i <= p.y + t; ++i) { uchar *g = path.ptr<uchar>(i); for (int j = p.x - t; j <= p.x + t; ++j) { if (g[j] == 255) { p.x = j; p.y = i; find = true; break; } } if (find) break; } if (find) { c.push_back(src.at<uchar>(p.y, p.x)); path.at<uchar>(p.y, p.x) = 0; wn = WinDataNum(path, p, size); } else break; } return c; } // 獲取窗口內(nèi)的有效數(shù)據(jù)個(gè)數(shù) int WinDataNum(cv::Mat path, cv::Point p, int size) { int number = 0; int t = size / 2; for (int i = p.y - t; i <= p.y + t; ++i) { uchar *g = path.ptr<uchar>(i); for (int j = p.x - t; j <= p.x + t; ++j) { if (g[j] == 255) number++; } } return number; }
測(cè)試效果
圖1 原圖
圖2 掩膜內(nèi)圖像
如圖1圖2所示,掩膜內(nèi)的圖像數(shù)據(jù)就是我們要提取的目標(biāo)。
圖3 放大后數(shù)據(jù)搜索路徑
圖3放大后可以看出,起點(diǎn)是230,之后的數(shù)據(jù)是230、231、236、232、234、146等等,再看c容器中的數(shù)據(jù)。
圖4 容器內(nèi)數(shù)據(jù)
對(duì)比完開(kāi)頭,再看結(jié)尾,如圖3所示,是234、234、231、234、234,然后就是起點(diǎn)230,查看容器。
圖5 容器內(nèi)數(shù)據(jù)
這樣有的小伙伴可能覺(jué)得中間會(huì)不會(huì)有數(shù)據(jù)錯(cuò)誤呢,很簡(jiǎn)單,打開(kāi)VS復(fù)制代碼后,搭配ImageWatch插件,debug調(diào)試打斷點(diǎn)觀察path矩陣,看看它的255數(shù)據(jù)是不是按預(yù)想的路徑消失,如果是則說(shuō)明扔的數(shù)據(jù)也沒(méi)有問(wèn)題。
總結(jié)
本文提供的只是一個(gè)簡(jiǎn)單思路,有一定局限性。比如該方法在圓線寬為1時(shí)效果最佳,若線寬大了就不能用窗口簡(jiǎn)單判斷了;另外,起點(diǎn)在右側(cè)時(shí)是逆時(shí)針獲取數(shù)據(jù),起點(diǎn)在左側(cè)時(shí)是順時(shí)針獲取數(shù)據(jù),如果想統(tǒng)一標(biāo)準(zhǔn)的話,最好加上起點(diǎn)的位置判斷,然后決定是否將c的數(shù)據(jù)翻轉(zhuǎn)。至于運(yùn)行速度方面,3000*3000的圖像矩陣中運(yùn)行基本為0ms,畢竟只是提取了一條圓線而已。
到此這篇關(guān)于OpenCV提取圖像中圓線上的數(shù)據(jù)具體流程的文章就介紹到這了,更多相關(guān)OpenCV提取圖像數(shù)據(jù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++設(shè)計(jì)類(lèi)不能被繼承的方法實(shí)例講解
在Java 中定義了關(guān)鍵字final,被final修飾的類(lèi)不能被繼承,C++中如何實(shí)現(xiàn),下面我們來(lái)看一個(gè)例子2013-12-12大數(shù)據(jù)情況下桶排序算法的運(yùn)用與C++代碼實(shí)現(xiàn)示例
在排序元素很多的情況下,其實(shí)桶排序的性能并不是太高,這里我們配合單鏈表的直接插入排序,來(lái)看下一大數(shù)據(jù)情況下桶排序算法的運(yùn)用與C++代碼實(shí)現(xiàn)示例:2016-07-07C++中new與delete、malloc與free應(yīng)用分析
這篇文章主要介紹了C++中new與delete、malloc與free應(yīng)用分析,很重要的概念,需要的朋友可以參考下2014-08-08