OpenCV:查找二进制Mat图像的所有非零坐标

DMo*_*Mor 19 c++ computer-science opencv image-processing computer-vision

我正试图找到二进制图像的非零(x,y)坐标.

我发现了一些函数的引用countNonZero(),它只计算非零坐标,findNonZero()而且我不确定如何访问或使用它,因为它似乎完全从文档中删除了.

是我发现的最接近的参考,但仍然没有任何帮助.我将不胜感激任何具体的帮助.

编辑: - 要指定,这是使用C++

Wan*_*ong 33

以下是如何findNonZero()保存非零元素的说明.以下代码对于访问二进制图像的非零坐标非常有用.方法1 findNonZero()在OpenCV中使用,方法2检查每个像素以找到非零(正)的像素.

方法1:

#include <iostream>
#include <opencv2/core/core.hpp>
#include <opencv2/highgui/highgui.hpp>
using namespace std;
using namespace cv;

int main(int argc, char** argv) {
    Mat img = imread("binary image");
    Mat nonZeroCoordinates;
    findNonZero(img, nonZeroCoordinates);
    for (int i = 0; i < nonZeroCoordinates.total(); i++ ) {
        cout << "Zero#" << i << ": " << nonZeroCoordinates.at<Point>(i).x << ", " << nonZeroCoordinates.at<Point>(i).y << endl;
    }
    return 0;
}
Run Code Online (Sandbox Code Playgroud)

方法2:

#include <iostream>
#include <opencv2/core/core.hpp>
#include <opencv2/imgproc/imgproc.hpp>
#include <opencv2/highgui/highgui.hpp>
using namespace std;
using namespace cv;

int main(int argc, char** argv) {
    Mat img = imread("binary image");
    for (int i = 0; i < img.cols; i++ ) {
        for (int j = 0; j < img.rows; j++) {
            if (img.at<uchar>(j, i) > 0) {  
                cout << i << ", " << j << endl;     // Do your operations
            }
        }
    }
    return 0;
}
Run Code Online (Sandbox Code Playgroud)

  • 循环顺序应该颠倒:遍历行,然后遍历列.OpenCV的Mat容器以行主顺序存储,因此迭代列对缓存不友好. (3认同)

Rog*_*and 5

为OpenCV 2.4.3提供了以下源代码,这可能会有所帮助:

#include <opencv2/core/core.hpp>
#include <vector>

/*! @brief find non-zero elements in a Matrix
 *
 * Given a binary matrix (likely returned from a comparison
 * operation such as compare(), >, ==, etc, return all of
 * the non-zero indices as a std::vector<cv::Point> (x,y)
 *
 * This function aims to replicate the functionality of
 * Matlab's command of the same name
 *
 * Example:
 * \code
 *  // find the edges in an image
 *  Mat edges, thresh;
 *  sobel(image, edges);
 *  // theshold the edges
 *  thresh = edges > 0.1;
 *  // find the non-zero components so we can do something useful with them later
 *  vector<Point> idx;
 *  find(thresh, idx);
 * \endcode
 *
 * @param binary the input image (type CV_8UC1)
 * @param idx the output vector of Points corresponding to non-zero indices in the input
 */
void find(const cv::Mat& binary, std::vector<cv::Point> &idx) {

    assert(binary.cols > 0 && binary.rows > 0 && binary.channels() == 1 && binary.depth() == CV_8U);
    const int M = binary.rows;
    const int N = binary.cols;
    for (int m = 0; m < M; ++m) {
        const char* bin_ptr = binary.ptr<char>(m);
        for (int n = 0; n < N; ++n) {
            if (bin_ptr[n] > 0) idx.push_back(cv::Point(n,m));
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

注意 - 看起来函数签名是错误的,所以我将输出更改vector为pass-by-reference.