OpenCV: найдите маленькие темные (черные) точки внутри круга - PullRequest
2 голосов
/ 12 марта 2020

Я пытаюсь определить круги, которые являются черными точками или имеют черные точки в них (те, которые я указал стрелкой на следующем изображении). enter image description here

Мой текущий подход заключается в использовании функции HoughCircles в OpenCV для обнаружения кругов с радиусом более 2 пикселей. Мой вопрос к сообществу: скажем, я обнаружил эти круги (как показано ниже), как я могу отделить круги, которые я указал стрелками , от остальных. Круги, на которые указывает стрелка, - это круги, которые меня интересуют, те, которые имеют черный / темный цвет. Розовые блики - это те, которые обнаружил HoughCircles. Кроме того, я сам добавил стрелки, чтобы показать круг моих интересов.

enter image description here

Вот код python, которым я был с помощью. Вот ссылка на изображение на случай, если вы захотите его попробовать.

import sys
import cv2 as cv
import numpy as np
import math
filename = '72471_125_df.jpg'
src = cv.imread("72471_125_df.jpg", cv.IMREAD_COLOR)
gray = cv.cvtColor(src, cv.COLOR_BGR2GRAY)
gray = cv.medianBlur(gray, 3)
# Inverse ratio of the accumulator resolution to the image resolution. For example, if dp=1 , the accumulator has the same resolution as the input image. If dp=2 , the accumulator has half as big width and height.
accum_size = 1
# Minimum distance between the centers of the detected circles.
minDist = 30
#First method-specific parameter. In case of CV_HOUGH_GRADIENT , it is the higher threshold of the two passed to the Canny() edge detector (the lower one is twice smaller).
param1 = 50
# Second method-specific parameter. In case of CV_HOUGH_GRADIENT , it is the accumulator threshold for the circle centers at the detection stage. The smaller it is, the more false circles may be detected. Circles, corresponding to the larger accumulator values, will be returned first.
param2 = 5
#
minRadius = 1
# 
maxRadius = 10
circles = cv.HoughCircles(gray, cv.HOUGH_GRADIENT, accum_size, minDist,
                           param1=param1, param2=param2,
                           minRadius=minRadius, maxRadius=maxRadius)
circles = circles.reshape(1,circles.shape[1], circles.shape[2])
if circles is not None:
    circles = np.uint16(np.around(circles))
    for ind, i in enumerate(circles[0, :]):
        center = (i[0], i[1])
        radius = 15
        cv.circle(src, center, radius, (255, 0, 255), 3)

cv.imwrite("modif_"+filename,src)

Обратите внимание, что я использовал конфигурацию, в которой minRadius = 2 и maxRadius = 5, однако она это сделала не дай мне круги с черными / темными точками в них. По какой-то причине он возвращает и другие круги. Кроме того, я попробовал подходы пороговой обработки, однако это не было устойчиво при всех условиях освещения. Пожалуйста, обратитесь к этой ссылке для просмотра изображений с различными условиями освещения.

1 Ответ

0 голосов
/ 12 марта 2020

Это подход, который я выбрал и который может быть использован для вдохновения. Я не уверен, что он охватывает все случаи "черных точек", но вы можете судить об этом. Я просто добавил порог к загруженному изображению, а затем повторно использовал предоставленный вами код.

import cv2
import numpy as np

img = cv2.imread('blackdots.jpg')
gray_img = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)

ret, thresh = cv2.threshold(gray_img, 170, 255, cv2.THRESH_BINARY) # <--- Try different values here

accum_size = 1
# Minimum distance between the centers of the detected circles.
minDist = 30
#First method-specific parameter. In case of CV_HOUGH_GRADIENT , it is the higher threshold of the two passed to the Canny() edge detector (the lower one is twice smaller).
param1 = 50
# Second method-specific parameter. In case of CV_HOUGH_GRADIENT , it is the accumulator threshold for the circle centers at the detection stage. The smaller it is, the more false circles may be detected. Circles, corresponding to the larger accumulator values, will be returned first.
param2 = 5
#
minRadius = 1
#
maxRadius = 10
circles = cv2.HoughCircles(thresh, cv2.HOUGH_GRADIENT, accum_size, minDist,
                           param1=param1, param2=param2,
                           minRadius=minRadius, maxRadius=maxRadius)
circles = circles.reshape(1,circles.shape[1], circles.shape[2])
if circles is not None:
    circles = np.uint16(np.around(circles))
    for ind, i in enumerate(circles[0, :]):
        center = (i[0], i[1])
        radius = 15
        cv2.circle(img, center, radius, (255, 0, 255), 3)

thresh = cv2.resize(thresh, (1280, 720)) # <---- This is just for easier display
img = cv2.resize(img, (1280, 720)) # <---- This is just for easier display
cv2.imwrite('circles_black_dot.png', img)
cv2.imwrite('threshold_black_dots.png', thresh)

Пороговое изображение:

enter image description here

Исходное изображение с кругами:

enter image description here

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...