有没有办法使用opencv检测图像中的相邻圆圈

Posted

技术标签:

【中文标题】有没有办法使用opencv检测图像中的相邻圆圈【英文标题】:Is there any way to detect adjacent circles in image using opencv 【发布时间】:2016-03-22 12:24:02 【问题描述】:

我正在尝试检测图像中的相邻圆圈。这些可以是4或5。有什么方法可以在opencv中检测到它。我尝试了很多方法,包括霍夫圆法。但我也在检测额外的圆圈。如果在任何情况下我能够检测到圆形,那么相同的参数将不适用于其他图像。

请告诉我任何可能实现此目的的方法。

我使用霍夫圆的代码是:

Mat img, gray;
img = imread("/Users/Development/Desktop/Images/IMG_0297.jpg");
cvtColor(img, gray, CV_BGR2GRAY);
// smooth it, otherwise a lot of false circles may be detected
GaussianBlur( gray, gray, Size(9, 9), 2, 2 );
vector<Vec3f> circles;
HoughCircles(gray, circles, CV_HOUGH_GRADIENT, 2, gray.rows/16,80,100,30,50 );
for( size_t i = 0; i < circles.size(); i++ )

    Point center(cvRound(circles[i][0]), cvRound(circles[i][1]));
    int radius = cvRound(circles[i][2]);
    // draw the circle center
    circle( img, center, 3, Scalar(0,255,0), -1, 8, 0 );
    // draw the circle outline
    circle( img, center, radius, Scalar(0,0,255), 3, 8, 0 );

namedWindow( "circles", 1 );
imshow( "circles", img );
waitKey(0);
return 0;

示例图片是

我想检测其中相邻的表盘

【问题讨论】:

【参考方案1】:

您可以使用partition 对相邻圆的圆进行聚类,即中心距离与其半径的 sim 相似的圆。您只需要定义适当的等价谓词,这里在CirclesOnSameLine 中实现。您最终可以改进此谓词,以仅将具有相似半径的圆视为相等。

这种聚类的结果是这样的(相同的颜色意味着相同的聚类):

使用这种方法,您可以安全地检测到一些圆,因为您可以删除不属于超过 4-5 个圆的集群的圆。

代码:

#include <opencv2/opencv.hpp>
#include <vector>
using namespace cv;
using namespace std;

struct CirclesOnSameLine

    float _tolerance;

    CirclesOnSameLine(float tolerance) : _tolerance(tolerance) ;

    bool operator()(const Vec3f& lhs, const Vec3f& rhs)
    
        // [0] = x
        // [1] = y
        // [2] = radius
        float center_distance = sqrt((lhs[0] - rhs[0])*(lhs[0] - rhs[0]) + (lhs[1] - rhs[1])*(lhs[1] - rhs[1]));
        float sum_radii = lhs[2] + rhs[2];

        if (sum_radii > center_distance)
        
            return (sum_radii / center_distance) < _tolerance;
        
        return (center_distance / sum_radii) < _tolerance;
    
;


int main()

    Mat3b img = imread("path_to_image");

    Mat1b gray;
    cvtColor(img, gray, COLOR_BGR2GRAY);

    GaussianBlur(gray, gray, Size(9, 9), 2, 2);
    vector<Vec3f> circles;
    HoughCircles(gray, circles, CV_HOUGH_GRADIENT, 2, gray.rows / 16, 80, 100, 10, 100);

    // Cluster circles near each other
    vector<int> labels;
    int n_labels = partition(circles, labels, CirclesOnSameLine(1.1f));

    vector<Scalar> colors;
    for (int i = 0; i < n_labels; ++i)
    
        Scalar color(rand() & 255, rand() & 255, rand() & 255);
        colors.push_back(color);
    

    Mat3b adjacent = img.clone();
    for (size_t i = 0; i < circles.size(); i++)
    
        Point center(cvRound(circles[i][0]), cvRound(circles[i][1]));
        int radius = cvRound(circles[i][2]);
        // draw the circle outline
        circle(adjacent, center, radius, colors[labels[i]], 3, 8, 0);
    

    // Remove small clusters
    vector<int> count(labels.size(), 0);
    for (size_t i = 0; i < labels.size(); ++i)
    
        count[labels[i]]++;
    

    Mat3b big_clusters = img.clone();
    for (size_t i = 0; i < circles.size(); i++)
    
        if (count[labels[i]] < 4) continue;

        Point center(cvRound(circles[i][0]), cvRound(circles[i][1]));
        int radius = cvRound(circles[i][2]);
        // draw the circle outline
        circle(big_clusters, center, radius, Scalar(0, 0, 255), 3, 8, 0);
    



    imshow("Adjacent circles", adjacent);
    imshow("Adjacent circles", big_clusters);

    waitKey();

    return 0;

【讨论】:

【参考方案2】:

例如,您可以对圆进行倒角匹配,然后使用边缘检测和图像扫描检查圆是否与另一个圆共享边缘或靠近它,以查看圆是否足够接近以相邻.

使用这个特定的图像,您可能可以做 kmeans 和连接的组件。然后倒角匹配圆,看看一个连通的组件是否由多个圆组成。

【讨论】:

opencv 不支持倒角匹配。这是参考code.opencv.org/issues/3602

以上是关于有没有办法使用opencv检测图像中的相邻圆圈的主要内容,如果未能解决你的问题,请参考以下文章

检测图像中的多个圆圈

从纸张目标中检测圆圈和镜头

在 OpenCV 中检测半圆

图像处理Opencv Python中的角点检测

OpenCV:创建一个透明蒙版?

opencv中的HoughCircles函数可以检测圆圈内的圆圈吗?