将unsigned char * image转换为OpenCV Mat的问题

时间:2013-10-04 01:11:36

标签: c++ opencv image-processing computer-vision mat

我有一个问题是将图像数据从帧抓取器解析为OpenCV Mat格式。我可以从我的EDT帧抓取器获取图像数据作为无符号字符指针并将其传递给新创建的Mat,但我在此过程中丢失了有价值的数据,我不知道如何修复它。

我的相机是红外相机,可输出12位320x256拜耳图像。我相信我使用EDT的API是正确的。使用EDT的软件“pdvshow”,我可以按预期查看图像数据,但是当我将返回的帧从EDT的API转换为OpenCV Mat并显示它时,我丢失了大量数据。当我将Mat设置为CV16UC1时,框架接近黑色,当Mat设置为CV8UC1时,框架显示大多数数据但看起来非常颗粒状,并且一些斑点完全被吹灭。我知道帧抓取器将每个12位像素存储在两个字节中,并且数据是MSB对齐的。 GetStride返回0。

unsigned char *pdvImage;
pdvImage = pdv_image(pdv_p); 

cv::Mat freshFrame;
freshFrame = cv::Mat(GetHeight(), GetWidth(), CV_16UC1, pdvImage, GetStride()); //was CV_16UC1 but 8UC1 shows more data

return freshFrame.clone();

2 个答案:

答案 0 :(得分:2)

我编写了一个代码,用于绘制char *图像,然后将其转换为OpenCV Mat:

const int WIDTH = 800, HEIGHT = 600;

uchar *data = new uchar[WIDTH * HEIGHT * 3];

for (int i = 0; i < WIDTH*HEIGHT*3;i+=3)
{
    if (i < WIDTH*HEIGHT*3/2)
    {
        data[i + 0] = (uchar)0;
        data[i + 1] = (uchar)0;
        data[i + 2] = (uchar)255;
    }
    else
    {
        data[i + 0] = (uchar)255;
        data[i + 1] = (uchar)0;
        data[i + 2] = (uchar)0;
    }
}

cout << "May God be pleased with you, amen!\n";

Mat colorfrm = Mat(HEIGHT, WIDTH, CV_8UC3);
colorfrm.data = data;

while (1)
{
    imshow("Original Image", colorfrm);

    /////////

    int c = cvWaitKey(30);
    if (c == ' ')
    {
        break;
    }
    if (c == 'q' || c == 'Q' || c == 27)
    {
        return 0;
    }
}

我希望这会有所帮助!

答案 1 :(得分:1)

当您创建freshFrame作为CV_16UC1 Mat时,12位数据位于最低有效位。缩放数据将解决产生的图像显得太黑的问题。而不是

return freshFrame.clone();

你需要

return freshFrame * 16;

您可能还需要检查pdvImage中每个像素的两个字节的字节顺序。

关于CV_8UC1,这将不起作用

freshFrame = cv::Mat(GetHeight(), GetWidth(), CV_8UC1, pdvImage, GetStride()); 

因为pdvImage的前两个字节构成了图像中第一个像素的值,但却成为了freshFrame中的两个独立像素。但是,如果您在上面的行中创建freshFrame,则第一个字节将成为freshFrame中第一个像素的值,第二个字节将成为第二个像素的值,依此类推。显然这是错误的。如果你真的需要一个CV_8UC1 Mat,最简单的可能就是创建一个CV_16UC1 Mat然后转换它。