最近一直有人向我提問很多二值圖像分析相關(guān)的問題,特別選擇了兩個(gè)典型的輪廓分析問題。進(jìn)行分析與編碼實(shí)現(xiàn)與演示,廢話不多說,先看第一個(gè)問題。
問題一
描述如下:
想找到工具盤中間缺少的幾個(gè)點(diǎn),統(tǒng)計(jì)出可以看到的工件數(shù)目
仔細(xì)分析圖像發(fā)現(xiàn),中間都毫無另外的有個(gè)白色很亮的圓圈,這個(gè)給了我兩個(gè)思路
可以通過霍夫變換檢測(cè)圓來提取到
可以通過二值圖像分析來提取 + 輪廓分析來提取到這些點(diǎn)
得到這些輪廓點(diǎn)之后通過分析整個(gè)輪廓區(qū)域得到傾斜角度,進(jìn)行糾偏,然后通過X與Y投影進(jìn)行分割,得到每個(gè)零件的中心位置坐標(biāo),根據(jù)每一行的間隔設(shè)置閾值,從而實(shí)現(xiàn)缺少部分部分的標(biāo)出與件數(shù)統(tǒng)計(jì),確定了這樣的思路以后,我就開始了寫代碼。代碼實(shí)現(xiàn)是基于輪廓分析的思路,因?yàn)檫@個(gè)方法,用的閾值比較少,有利于算法穩(wěn)定性檢測(cè)。演示各部輸出。二值化處理之后(形態(tài)學(xué)處理):
輪廓發(fā)現(xiàn)與校正角度之后
投影分析與統(tǒng)計(jì)結(jié)果如下:
此外基于霍夫也是可以嘗試的,霍夫的二值化效果也比較好,顯示如下:
感興趣的同學(xué)可以自己繼續(xù)嘗試下去。
問題二
描述如下:
如何統(tǒng)計(jì)下圖中的對(duì)象個(gè)數(shù),原圖如下
看到這個(gè)圖像之后,個(gè)人覺得解決十分簡單,基于最外層輪廓發(fā)現(xiàn)即可,無需樹形結(jié)構(gòu)與層次分析,集合圖像形態(tài)學(xué)分析或者距離變換就可以得到,最終代碼的運(yùn)行結(jié)果如下:
代碼
問題1的代碼如下(已經(jīng)添加各步驟注釋了):
src=cv.imread("D:/images/zsxq/zsxq_01.jpg") cv.imshow("input",src) #二值化處理 gray=cv.cvtColor(src,cv.COLOR_BGR2GRAY) se=cv.getStructuringElement(cv.MORPH_RECT,(3,3)) dst=cv.morphologyEx(gray,cv.MORPH_GRADIENT,se) ret,binary=cv.threshold(dst,0,255,cv.THRESH_OTSU|cv.THRESH_BINARY) #形態(tài)學(xué)處理 se=cv.getStructuringElement(cv.MORPH_ELLIPSE,(5,5)) binary=cv.morphologyEx(binary,cv.MORPH_OPEN,se) se=cv.getStructuringElement(cv.MORPH_ELLIPSE,(10,10)) binary=cv.morphologyEx(binary,cv.MORPH_CLOSE,se) cv.imshow("binary",binary) #輪廓分析 contours,hireachy=cv.findContours(binary,cv.RETR_EXTERNAL,cv.CHAIN_APPROX_SIMPLE) result=np.zeros_like(binary) total=0 forcntinrange(len(contours)): area=cv.contourArea(contours[cnt]) ifarea55: ????????continue ????rrt?=?cv.minAreaRect(contours[cnt]) ????cx,?cy?=?rrt[0] ????cv.circle(result,?(np.int32(cx),?np.int32(cy)),?5,?(255),?-1) ????total?+=?1 #?幾何糾偏 h,?w?=?result.shape pts?=?[] for?row?in?range(h): ????for?col?in?range(w): ????????pv?=?result[row,?col] ????????if?pv?==?255: ????????????pts.append((col,?row)) pts?=?np.array(pts) rrt?=?cv.minAreaRect(pts) print(rrt) M?=?cv.getRotationMatrix2D(((w-1)/2.0,(h-1)/2.0),rrt[2],1) dst?=?cv.warpAffine(result,M,(w,h)) src?=?cv.warpAffine(src,M,(w,h)) #?Y方向投影 tbins?=?y_split(dst) #?X?方向投影 print("y-step",?tbins) for?i?in?range(0,?len(tbins),?1): ????if?i?==?0: ????????roi?=?dst[0:tbins[i],?0:w] ????????src_roi?=?src[0:tbins[i],?0:w,?:] ????????x_projection(roi,?src_roi) ????????cv.imshow("roi",?roi) ????????cv.waitKey(0) ????if?i?==?len(tbins)-1: ????????roi?=?dst[tbins[i]:h-1,?0:w] ????????src_roi?=?src[tbins[i]:h-1,?0:w,?:] ????????x_projection(roi,?src_roi) ????????cv.imshow("roi",?roi) ????????cv.waitKey(0) ????if?0?
問題2的完整代碼如下:
importcv2ascv importnumpyasnp src=cv.imread("D:/images/zsxq/zsxq_02.jpg") cv.imshow("input",src) src=cv.GaussianBlur(src,(3,3),0) gray=cv.cvtColor(src,cv.COLOR_BGR2GRAY) edge=cv.Canny(src,50,100) se=cv.getStructuringElement(cv.MORPH_ELLIPSE,(10,10)) binary=cv.morphologyEx(edge,cv.MORPH_CLOSE,se) contours,hireachy=cv.findContours(binary,cv.RETR_EXTERNAL,cv.CHAIN_APPROX_SIMPLE) count=0 forcntinrange(len(contours)): area=cv.contourArea(contours[cnt]) ifarea100: ????????continue ????count?+=?1 ????rrt?=?cv.minAreaRect(contours[cnt]) ????#?rotated?rectangle ????box?=?cv.boxPoints(rrt) ????box?=?np.intp(box) ????cv.drawContours(src,?[box],?0,?(255,?0,?0),?2) cv.imshow("binary",?binary) cv.imshow("result",?src) cv.waitKey(0) cv.destroyAllWindows()
審核編輯:湯梓紅
-
算法
+關(guān)注
關(guān)注
23文章
4709瀏覽量
95362 -
代碼
+關(guān)注
關(guān)注
30文章
4900瀏覽量
70719 -
二值圖像
+關(guān)注
關(guān)注
0文章
14瀏覽量
8853
原文標(biāo)題:二值圖像分析案例精選
文章出處:【微信號(hào):CVSCHOOL,微信公眾號(hào):OpenCV學(xué)堂】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
發(fā)布評(píng)論請(qǐng)先 登錄
基于FPGA的二值圖像的膨脹算法的實(shí)現(xiàn)
基于圖像分塊的局部閾值二值化方法
一種有效的文本圖像二值化方法
巖石節(jié)理二值圖像的細(xì)化
基于大津法的圖像分塊二值化算法
基于Legendre矩的多閾值二值圖像重構(gòu)算法
基于邊界特征的圖像二值化方法應(yīng)用研究
二值圖像目標(biāo)鄰域點(diǎn)法邊界跟蹤算法

基于圖像分塊的局部閾值二值化方法

評(píng)論