drawContours 奇怪的行为?

drawContours strange behaviour?

我从二值图像中过滤小斑点(面积小于阈值的轮廓)。 掩码是二值图像。

如果我注释行

 drawContours(mask, contours, -1, Scalar(255), CV_FILLED, 8);

然后我在填充 0 个小斑点后保存蒙版时得到奇怪的结果。

也不明白为什么它在取消注释行时起作用,因为在

之后
drawContours(mask, contours, -1, Scalar(255), CV_FILLED, 8);

掩码逻辑上应该与输入掩码相同(图像周围的 1 个像素边界除外)

void FilterSmallBlobs(Mat &mask, float minArea)
{
    //as side effect this code extends inner holes with 1 pixel border and removes 1 pixels border from image border.

    vector<vector<Point>> contours;
    //findContours(mask, contours, CV_RETR_LIST, CV_CHAIN_APPROX_SIMPLE);
    findContours(mask, contours, CV_RETR_LIST, CV_CHAIN_APPROX_NONE);

    vector<vector<Point>> badContours; //contours to erase
    for (int i = 0; i < (int)contours.size(); i++)
    {
        if(contourArea(contours[i]) <= minArea)
            badContours.push_back(contours[i]);
    }

    //drawContours(mask, contours, -1, Scalar(255), CV_FILLED, 8);
    drawContours(mask, badContours, -1, Scalar(0), CV_FILLED, 8);
}

我得到了什么

我想要的

所以我不明白当我填充错误的轮廓时 drawContours 会破坏初始蒙版吗?

findContours

的文档中所述

Note: Source image is modified by this function.

因此,在您的情况下,您看到的是修改后图像的某些部分,而其他部分被覆盖,因为您将小斑点绘制为黑色。

这段代码应该阐明这一点:

#include <opencv2\opencv.hpp>
using namespace cv;

int main()
{
    Mat1b img = imread("path_to_image", IMREAD_GRAYSCALE);

    vector<vector<Point>> contours;

    Mat1b will_be_modified = img.clone();
    findContours(will_be_modified, contours, RETR_LIST, CHAIN_APPROX_SIMPLE);

    for (int i = 0; i < contours.size(); ++i)
    {
        if (contourArea(contours[i]) < 3000)
        {
            drawContours(img, contours, i, Scalar(0), CV_FILLED);
        }
    }

    imshow("img", img);
    imshow("After findContours", will_be_modified);
    waitKey();

    return 0;
}

结果:

图像传递给 findContours