【问题标题】:PCA project and backproject in OpencvOpencv 中的 PCA 项目和 backproject
【发布时间】:2012-04-11 23:58:58
【问题描述】:

我在 Ubuntu Opencv 中工作。我正在尝试对单个图像进行 PCA 分析。我采用 3 通道图像并将其更改为具有 3 列和 r*c 行数的单通道图像。r 和 c是原始图像的行和列。当我在 PCA 上进行反投影后尝试显示重建图像时,它给了我一个绿色图像。这是我的代码

Mat pcaset=cvCreateMat(image->height*image->width,image->nChannels,CV_8UC1);
for(int i=0;i<image->height;i++)
    {
        for(int j=0;j<image->width;j++)
        {
            for(int k=0;k<image->nChannels;k++)
            (ptrpcaset+i*pcaset.step)[k]=((ptrimage+i*image->widthStep)[3*j+k]);

       }

    }
int nEigens=3;
    Mat databackprojected;
    PCA pca(pcaset,Mat(),CV_PCA_DATA_AS_ROW,nEigens);
    Mat dataprojected(pcaset.rows,nEigens,CV_8UC1);
    pca.project(pcaset,dataprojected);
    pca.backProject(dataprojected,databackprojected);
    Mat backprojectnorm;//(databackprojected.rows,nEigens,CV_8UC1);
    normalize(databackprojected,backprojectnorm,0,255,NORM_MINMAX,-1);
    Mat finaldataafterreshaping(image->height,image->width,CV_8UC3);
    uchar* finalptr=(uchar*)finaldataafterreshaping.data;
    uchar* ptrnorm=(uchar*)backprojectnorm.data;

    int x=0,y=0,i=0;

    while(i<backprojectnorm.rows)
    {
        while(x<image->height)
        {
            while(y<image->width)
            {
                for(int k=0;k<image->nChannels;k++)
                {
                    (finalptr+x*finaldataafterreshaping.step)[3*y+k]=(ptrnorm+i*backprojectnorm.step)[k];
                }
                y=y+1;i=i+1;
            }
            x=x+1;y=0;
        }
    }
imshow("Reconstructed data",finaldataafterreshaping);

【问题讨论】:

    标签: c++ ubuntu opencv computer-vision pca


    【解决方案1】:

    您需要进行以下更改:

    (ptrpcaset+(j + i*image->width)*pcaset.step)[k]=((ptrimage+i*image->widthStep)[3*j+k]);
    

    因为您在转换数据时没有考虑 j 坐标,因此最后您只将图像的最后一行保存在新矩阵中。

    当你重塑你的数据时,你需要做这样的事情:

    float* val = (float*)&(ptrnorm+i*backprojectnorm.step)[(k*4)];
    (finalptr+x*finaldataafterreshaping.step)[3*y+k]=*val;
    

    因为您得到的矩阵类型为float 而不是uchar。所以你需要某种转换。我不确定这样做是否是个好主意,但它确实有效。我建议你看看 OpenCV 2 的 C++ API,它可以更好地处理这些事情。

    另外,不需要整个while(i&lt;backprojectnrom.rows) 循环。

    【讨论】:

    • 好吧,我正在尝试将 3 通道矩阵转换为 n*3 矩阵,其中每一行代表像素的 rgb 值。所以如果我这样做 (ptrpcaset+(j + i*image->width) *pcaset.step)[k] 这将如何工作
    • n 应该是width x height?目前,您正在使用变量i 寻址目标矩阵的行。但是i 只取 0 到 image-&gt;height 之间的值。所以你可以看到有些地方不对,因为你没有访问你创建的矩阵中的大量行。因此,您需要将变量j(j + i*image-&gt;width) 类似的东西考虑在内。使用此公式,您可以获得每对 (i,j) 的唯一索引。所以你最终每个像素只有一行。
    猜你喜欢
    • 2012-03-14
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2012-01-23
    • 1970-01-01
    • 2011-11-27
    • 2017-12-22
    • 1970-01-01
    相关资源
    最近更新 更多