提及来,我仍是挺惭愧的,去年四月份连续在51CTO博客发表了一波技术文章,后来由于本身太忙就慢慢忘记更新拉,前两天一看访问量竟然突破100+万拉,感谢各位的厚爱与支持,更以为51CTO博客改版以后真的是很棒!因此又来更新了!html
最近一直有人向我提问不少二值图像分析相关的问题,特别选择了两个典型的轮廓分析问题。进行分析与编码实现与演示,废话很少说,先看第一个问题。算法
问题描述以下app
想找到工具盘中间缺乏的几个点,统计出能够看到的工件数目ide
仔细分析图像发现,中间都毫无另外的有个白色很亮的圆圈,这个给了我两个思路工具
- 能够经过霍夫变换检测圆来提取到
- 能够经过二值图像分析来提取 + 轮廓分析来提取到这些点
获得这些轮廓点以后经过分析整个轮廓区域获得倾斜角度,进行纠偏,而后经过X与Y投影进行分割,获得每一个零件的中心位置坐标,根据每一行的间隔设置阈值,从而实现缺乏部分部分的标出与件数统计,肯定了这样的思路之后,我就开始了写代码。代码实现是基于轮廓分析的思路,由于这个方法,用的阈值比较少,有利于算法稳定性检测。演示各部输出。二值化处理以后(形态学处理):
轮廓发现与校订角度以后学习
投影分析与统计结果以下编码
此外基于霍夫也是能够尝试的,霍夫的二值化效果也比较好,显示以下:
感兴趣的同窗能够本身继续尝试下去。code
描述以下:视频
如何统计下图中的对象个数,原图以下htm
看到这个图像以后,我的以为解决十分简单,基于最外层轮廓发现便可,无需树形结构与层次分析,集合图像形态学分析或者距离变换就能够获得,最终代码的运行结果以下:
问题1的代码以下(已经添加各步骤注释了):
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) # 形态学处理 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 for cnt in range(len(contours)): area = cv.contourArea(contours[cnt]) if area < 55: 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 < i < (len(tbins)-1): roi = dst[tbins[i-1]:tbins[i] - 1, 0:w] src_roi = src[tbins[i-1]:tbins[i] - 1, 0:w,:] x_projection(roi, src_roi) cv.imshow("roi", roi) cv.waitKey(0) # 显示结果 cv.imshow("result", result) cv.imshow("dst", dst) cv.putText(src, "numbers: " + str(total), (50, 50), cv.FONT_HERSHEY_SIMPLEX, 1.0, (0, 0, 255), 2) cv.imshow("detection", src) cv.waitKey(0) cv.destroyAllWindows()
问题二:
import cv2 as cv import numpy as np 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 for cnt in range(len(contours)): area = cv.contourArea(contours[cnt]) if area < 100: 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()
解锁更多图像处理与视频分析技能,
欢迎学习个人最新OpenCV4图像处理与视频分析视频课程
感谢你们关注个人博客,我会继续更新!有大家有动力!