使用OpenCV(C ++)显示Kinect流

时间:2015-01-21 16:50:56

标签: c++ opencv kinect kinect-sdk

我正在尝试将Kinect的RGB相机(使用SDK版本1.8)生成的流的每一帧都放到OpenCV(2.4.10)Mat_<Vec3b>中。这是我目前的算法,它一点也不快:

Mat_<Vec3b> mat = Mat::zeros(480, 640, CV_8UC3);

NUI_IMAGE_FRAME imageFrame;
NUI_LOCKED_RECT lockedRect;

if (sensor->NuiImageStreamGetNextFrame(colorStream, 0, &imageFrame) < 0) { return; }

INuiFrameTexture* texture = imageFrame.pFrameTexture;
texture->LockRect(0, &lockedRect, NULL, 0);


if (lockedRect.Pitch != 0)
{
    BYTE* upperLeftCorner = (BYTE*)lockedRect.pBits;
    BYTE* pointerToTheByteBeingRead = upperLeftCorner;

    for (int i = 0; i < 480; i++)
    {
        for (int j = 0; j < 640; j++)
        {
            unsigned char r = *pointerToTheByteBeingRead;
            pointerToTheByteBeingRead += 1;
            unsigned char g = *pointerToTheByteBeingRead;
            pointerToTheByteBeingRead += 1;
            unsigned char b = *pointerToTheByteBeingRead;
            pointerToTheByteBeingRead += 2; //So to skip the alpha channel

            mat.at<Vec3b>(Point(j, i))[0] = r;
            mat.at<Vec3b>(Point(j, i))[1] = g;
            mat.at<Vec3b>(Point(j, i))[2] = b;
        }
    }

}

texture->UnlockRect(0);
sensor->NuiImageStreamReleaseFrame(colorStream, &imageFrame);

我检查了the OpenCV documentation,我知道我应该使用指针访问来提高效率。 Mat_<Vec3b>以与Mat相同的方式存储到内存中,还是应该执行其他指针算法?

另外,我知道每次更新每个像素并不是通过Mat实现流显示的最有效方法。我还能做些什么?

1 个答案:

答案 0 :(得分:0)

最后想出了如何使用指针算法。代码不言自明:

Mat_<Vec3b> mat = Mat::zeros(480, 640, CV_8UC3);

NUI_IMAGE_FRAME imageFrame;
NUI_LOCKED_RECT lockedRect;

if (sensor->NuiImageStreamGetNextFrame(colorStream, 0, &imageFrame) < 0) { return; }

INuiFrameTexture* texture = imageFrame.pFrameTexture;
texture->LockRect(0, &lockedRect, NULL, 0);


if (lockedRect.Pitch != 0)
{
    BYTE* upperLeftCorner = (BYTE*)lockedRect.pBits;
    BYTE* pointerToTheByteBeingRead = upperLeftCorner;

    for (int i = 0; i < 480; i++)
    {
        Vec3b *pointerToRow = mat.ptr<Vec3b>(i);

        for (int j = 0; j < 640; j++)
        {
            unsigned char r = *pointerToTheByteBeingRead;
            pointerToTheByteBeingRead += 1;
            unsigned char g = *pointerToTheByteBeingRead;
            pointerToTheByteBeingRead += 1;
            unsigned char b = *pointerToTheByteBeingRead;
            pointerToTheByteBeingRead += 2; //So to skip the alpha channel

            pointerToRow[j] = Vec3b(r, g, b);
        }
    }

}

texture->UnlockRect(0);
sensor->NuiImageStreamReleaseFrame(colorStream, &imageFrame);