使用坐标Mat(OpenCV / C ++)访问图像值

时间:2016-01-25 11:19:19

标签: c++ opencv coordinates mat

在openCV(C ++)中,我有一个图像 IMG ,我想在 coor 中的位置存储中提取每个像素的值。是否有可能以有效的方式获得此像素值? foor循环(使用 .at(i,j))是否有效?是否有内置函数来执行此操作? 这是我的代码:

cv::Mat IMG = cv::imread("image.png",CV_LOAD_IMAGE_GRAYSCALE);
cv::Mat coor = (cv::Mat_<float>(3,2) << 1, 0, 0, 1, 2, 2); // search coordinates
cv::Mat pixel_values;
// and I'd like to do something like this:
//This should result in a matrix with the same size as *coor*
pixel_values = IMG.at(coor); // similar to the matrix accesing method in Matlab. 

2 个答案:

答案 0 :(得分:3)

可以通过不同的方式访问cv :: Mat中“像素”的值。您是否阅读过OpenCV的文档?

我建议你从表格here

开始

修改 我已经更正了代码:此代码适用于msvc 2015和opencv(3.1),但也包括&gt; 2是好的

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

#if defined(NDEBUG)
#pragma comment(lib, "opencv_world310.lib")
#else
#pragma comment(lib, "opencv_world310d.lib")
#endif // 


cv::Mat GetPixelsFromMat( const cv::Mat& I, const std::vector<cv::Point2f>& points )
{
    // some pre-condition:
    cv::Mat res( 1, (int)points.size( ), CV_8UC1 );

    int i = 0;
    for( const auto& point : points )
        res.ptr( 0 )[ i++ ] = I.at<uchar>( cvRound( point.y ), cvRound( point.x ) );

    return res;
}

int main( int argc, char* argv[] )
{
    cv::Mat testImg( 1, 10, CV_8UC1 );
    for( int i = 0; i < 10; ++i )
        testImg.ptr( 0 )[ i ] = i;

    std::vector<cv::Point2f> points;
    points.push_back( cv::Point2f( 1, 0 ) );
    points.push_back( cv::Point2f( 5, 0 ) );
    points.push_back( cv::Point2f( 9, 0 ) );

    cv::Mat pixelsMap = GetPixelsFromMat( testImg, points );

    if( pixelsMap.ptr( 0 )[ 0 ] == 1 )
        cout << "OK 0" << endl;
    else
        cout << "FAIL 0" << endl;

    if( pixelsMap.ptr( 0 )[ 1 ] == 5 )
        cout << "OK 1" << endl;
    else
        cout << "FAIL 1" << endl;

    if( pixelsMap.ptr( 0 )[ 2 ] == 9 )
        cout << "OK 2" << endl;
    else
        cout << "FAIL 2";


    return EXIT_SUCCESS;
}

这是一个很大的简化,但我认为这是一个起点。

答案 1 :(得分:0)

如果您已经知道矩阵的类型,可以轻松使用@ elvis.dukaj answer。您只需迭代坐标并检索相应的像素值。

但是,如果你需要更通用的东西支持:

  • 任何类型和渠道的输入矩阵
  • 坐标输入列表为cv::Matstd::vector,坐标为任意类型(intfloat等等。)

您可能会发现此getPixels函数很有用:

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

Mat getPixels(InputArray _src, InputArray _coords)
{
    // Get src matrix
    Mat src = _src.getMat();

    // Get coords matrix, eventually convert to integer type
    Mat coords = _coords.getMat();
    if (coords.type() != CV_32S)
    {
        coords.convertTo(coords, CV_32S);
    }
    // Check coordinates
    int n = coords.checkVector(2, CV_32S);
    CV_Assert(n > 0);

    // Create output matrix
    Mat dst(n, 1, src.type());

    // Get pixel values at given coordinates
    int esz = src.elemSize();
    Point* coords_ptr = (Point*)coords.data;
    uchar* dst_ptr = dst.data;

    for (int i = 0; i < n; i++, dst_ptr += esz)
    {
        memcpy(dst_ptr, src.ptr(coords_ptr[i].y) + coords_ptr[i].x * esz, esz);
    }

    return dst;
}


int main() 
{

    Mat img(5, 5, CV_8UC1);
    randu(img, Scalar(0, 0, 0), Scalar(10, 10, 10));

    vector<Point> coor{ Point(1, 0), Point(0, 1), Point(2, 2) };
    //cv::Mat coor = (cv::Mat_<float>(3, 2) << 1, 0, 0, 1, 2, 2);

    Mat out = getPixels(img, coor);

    cout << out;

    return 0;
}