使用 OpenCV 仅检测图像中的虚线(虚线)

Detect dotted (broken) lines only in an image using OpenCV

我正在尝试学习图像特征检测技术。

我已经设法检测到水平线(unbroken/continuous),但是我无法检测图像中的所有 dotted/broken 线。

这是我的测试图像,你可以看到有虚线和一些 text/boxes 等等

到目前为止,我使用了以下代码,它只检测到一条虚线。

import cv2
import numpy as np

img=cv2.imread('test.jpg')
img=functions.image_resize(img,1000,1000) #function from a script to resize image to fit my screen
imgGray=cv2.cvtColor(img,cv2.COLOR_BGR2GRAY)
imgEdges=cv2.Canny(imgGray,100,250)
imgLines= cv2.HoughLinesP(imgEdges,2,np.pi/100,60, minLineLength = 10, maxLineGap = 100)
for x1,y1,x2,y2 in imgLines[0]:
    cv2.line(img,(x1,y1),(x2,y2),(0,255,0),2)

cv2.imshow('Final Image with dotted Lines detected',img) 

我的输出图像如下。如您所见,我只设法检测到最后一条虚线。我玩过参数 rho,theta,min/max 行,但运气不好。

非常感谢任何建议:)

如果您对点的大小有想法,可以使用黑帽变换来过滤掉虚线。黑帽是图像和图像之间的差异。那你可以试试霍夫线变换。

所以,试试

将背景色转换为灰色

使用 morphologyEx 应用黑帽:这将只在生成的图像中留下黑点。

反转结果并尝试霍夫线变换。

在这里,您将不得不尝试使用内核大小来仅过滤点。如果证明这不是很可靠,另一种方法是使用斑点检测器。反转图像并应用 opencv blob 检测器或查找轮廓。按区域筛选 blobs/contours。字母和其他结构的面积将比点大,因此您可以删除任何比点大的结构。然后应用霍夫线变换。

这个解决方案:

import cv2
import numpy as np

img=cv2.imread('test.jpg')

kernel1 = np.ones((3,5),np.uint8)
kernel2 = np.ones((9,9),np.uint8)

imgGray=cv2.cvtColor(img,cv2.COLOR_BGR2GRAY)
imgBW=cv2.threshold(imgGray, 230, 255, cv2.THRESH_BINARY_INV)[1]

img1=cv2.erode(imgBW, kernel1, iterations=1)
img2=cv2.dilate(img1, kernel2, iterations=3)
img3 = cv2.bitwise_and(imgBW,img2)
img3= cv2.bitwise_not(img3)
img4 = cv2.bitwise_and(imgBW,imgBW,mask=img3)
imgLines= cv2.HoughLinesP(img4,15,np.pi/180,10, minLineLength = 440, maxLineGap = 15)

for i in range(len(imgLines)):
    for x1,y1,x2,y2 in imgLines[i]:
        cv2.line(img,(x1,y1),(x2,y2),(0,255,0),2)

cv2.imshow('Final Image with dotted Lines detected', img)