如何计算二进制图像上的白色物体?

Posted

技术标签:

【中文标题】如何计算二进制图像上的白色物体?【英文标题】:How to count white object on Binary Image? 【发布时间】:2016-05-31 08:13:22 【问题描述】:

我正在尝试从图像中计算对象。我使用日志照片,并使用一些步骤来获取二进制图像。

这是我的代码:

#include <opencv2/core/core.hpp>
#include <opencv2/imgproc/imgproc.hpp>
#include <opencv2/highgui/highgui.hpp>
#include <iostream>
#include <features2d.hpp>
using namespace cv;
using namespace std;
int main(int argc, char *argv[])

    //load image
    Mat img = imread("kayu.jpg", CV_LOAD_IMAGE_COLOR);
    if(img.empty())
       return -1;
    //namedWindow( "kayu", CV_WINDOW_AUTOSIZE );
    imshow("kayu", img);

    //convert to b/w
    Mat bw;
    cvtColor(img, bw, CV_BGR2GRAY);
    imshow("bw1", bw);

    threshold(bw, bw, 40, 255, CV_THRESH_BINARY);
    imshow("bw", bw);

    //distance transform & normalisasi
    Mat dist;
    distanceTransform(bw, dist, CV_DIST_L2, 3);
    normalize(dist, dist, 0, 2., NORM_MINMAX);
    imshow("dist", dist);

    //threshold to draw line
    threshold(dist, dist, .5, 1., CV_THRESH_BINARY);
    imshow("dist2", dist);

    //dist = bw;
    //dilasi
    Mat dilation, erotion, element;
    int dilation_type = MORPH_ELLIPSE;
    int dilation_size = 17;

    element = getStructuringElement(dilation_type, Size(2*dilation_size + 1, 2*dilation_size+1), Point(dilation_size, dilation_size ));
    erode(dist, erotion, element);
    int erotionCount = 0;
    for(int i=0; i<erotionCount; i++)
        erode(erotion, erotion, element);
    
    imshow("erotion", erotion);

    dilate(erotion, dilation, element);
    imshow("dilation", dilation);
    waitKey(0);
    return 0;

如您所见,我使用 Erosion 和 Dilation 来获得更好的日志圆形对象。我的问题是,我一直在数物体。我尝试了SimpleBlobDetector,但什么也没得到,因为当我尝试将“膨胀”步骤的结果转换为CV_8U 时,白色对象消失了。当我使用findContours() 时,我也遇到了错误。它说的是图像通道。我不能在这里显示错误,因为步骤太多,我已经从我的代码中删除了它。

顺便说一句,最后,我得到了 1 个图像通道。 我可以用它来计数,还是必须转换它?最好的方法是什么?

【问题讨论】:

OpenCV how to find a list of connected components in a binary image的可能重复 【参考方案1】:

两个简单的步骤:

    查找二值化图像的轮廓。 获取轮廓的计数。

代码:

int count_trees(const cv::Mat& bin_image)
    cv::Mat img;
    if(bin_image.channels()>1)
        cv::cvtColor(bin_image,img,cv::COLOR_BGR2GRAY);
    
    else
         img=bin_image.clone();;
    
    if(img.type()!=CV_8UC1)
        img*=255.f; //This could be stupid, but I do not have an environment to try it
        img.convertTo(img,CV_8UC1);
    

    std::vector<std::vector<cv::Point>> contours
    std::vector<Vec4i> hierarchy;
    cv::findContours( img, contours, hierarchy, CV_RETR_EXTERNAL, CV_CHAIN_APPROX_SIMPLE);
    return contours.size();

【讨论】:

感谢您的回复。我尝试了您的代码,但仍然收到有关不支持格式的错误。错误消息有点像“不支持的格式或格式组合,findContours() 仅支持 CV_8u”。 cvtColor(img, bw, CV_BGR2GRAY);阈值(体重,体重,40,255,CV_THRESH_BINARY);这些行确保您的图像是 CV_8UC1,因为它是作为正常颜色加载的。还有另一个导致该问题的代码。你应该告诉我们知道 我已经在问题中展示了我的所有代码。我也很想知道为什么会这样。我从您的代码中创建了新函数,输入参数是膨胀。 对不起!我只是注意到你在改变深度的距离变换之后使用它。我将编辑我的答案 轮廓查找是不必要的。使用连通分量分析可以有效地解决这个问题。 OpenCV 有一个“标记”算法可以做到这一点。【参考方案2】:

我有同样的问题,这是我即将实施的一个想法。

1) 将您的图像表示为整数数组; 0 = black, 1 = white.

2) 设置N = 2;

3) 逐像素扫描您的图像。每当你找到一个白色像素时,激活一个填充算法,从刚刚找到的像素开始;绘制值为N++的区域;

4) 迭代 3 直到到达最后一个像素。 (N-2) 是找到的区域数。

此方法取决于对象的形状;我的比你的更混乱(祝我好运……)。我将使用在某处找到的递归洪水填充配方(可能是 Rosetta Code)。

此解决方案还可以轻松计算每个区域的大小。

【讨论】:

这是连通分量分析。这是正确的做法。但是有更有效的算法。在 OpenCV 文档中搜索“label”。【参考方案3】:

尝试将其应用于您已删除的 img

// count
for (int i = 0; i< contours.size(); i = hierarchy[i][0]) // iteration sur chaque contour .

    Rect r = boundingRect(contours[i]);
    if (hierarchy[i][2]<0) 
        rectangle(canny_output, Point(r.x, r.y), Point(r.x + r.width, r.y + r.height), Scalar(20, 50, 255), 3, 8, 0);
        count++;
    

cout << "Numeber of contour = " << count << endl;
imshow("src", src);
imshow("contour", dst);
waitKey(0);

【讨论】:

以上是关于如何计算二进制图像上的白色物体?的主要内容,如果未能解决你的问题,请参考以下文章

通过分析二进制前景图像的白色像素来计算车辆计数所需的想法

使用ImageMagick计算PNG图像中的对象数

二进制图像上的快速像素计数 - ARM neon 内在函数 - iOS 开发

图像处理 - 计算二值图像中空白质心的算法

如何在 C++ 中使用 OpenCV 4.2 计算二进制图像的周长

数字图像基础