簡體   English   中英

在OpenCV 2.2中使用C ++語法的PCA + SVM

[英]PCA + SVM using C++ Syntax in OpenCV 2.2

我在使用Mat和PCA類使用最新的C ++語法時遇到了PCA和Eigenfaces的問題。 較舊的C語法將IplImage *數組作為參數來執行其處理,而當前API僅采用由Column或Row格式化的Mat。 我采用了Row方法,使用reshape函數來擬合我的圖像矩陣以適合單行。 我最終想要獲取這些數據然后使用SVM算法來執行檢測,但是當我這樣做時,我的所有數據都只是一個0流。 有人可以幫幫我嗎? 我究竟做錯了什么? 謝謝!

我看到了這個問題並且它有些相關,但我不確定解決方案是什么。

這基本上就是我所擁有的:

vector<Mat> images; //This variable will be loaded with a set of images to perform PCA on.
Mat values(images.size(), 1, CV_32SC1); //Values are the corresponding values to each of my images.

int nEigens = images.size() - 1; //Number of Eigen Vectors.

//Load the images into a Matrix
Mat desc_mat(images.size(), images[0].rows * images[0].cols, CV_32FC1);
for (int i=0; i<images.size(); i++) {
  desc_mat.row(i) = images[i].reshape(1, 1);
}

Mat average;
PCA pca(desc_mat, average, CV_PCA_DATA_AS_ROW, nEigens);

Mat data(desc_mat.rows, nEigens, CV_32FC1); //This Mat will contain all the Eigenfaces that will be used later with SVM for detection

//Project the images onto the PCA subspace
for(int i=0; i<images.size(); i++) {
  Mat projectedMat(1, nEigens, CV_32FC1);
  pca.project(desc_mat.row(i), projectedMat);

  data.row(i) = projectedMat.row(0);
}

CvMat d1 = (CvMat)data;
CvMat d2 = (CvMat)values;

CvSVM svm;
svm.train(&d1, &d2);
svm.save("svmdata.xml");

etarion說的是正確的。

要復制列或行,您必須始終寫:

Mat B = mat.col(i);
A.copyTo(B);

以下程序顯示了如何在OpenCV中執行PCA。 它將顯示平均圖像和前三個特征臉。 我在那里使用的圖像可以從http://www.cl.cam.ac.uk/research/dtg/attarchive/facedatabase.html獲得

#include "cv.h"
#include "highgui.h"

using namespace std;
using namespace cv;

Mat normalize(const Mat& src) {
    Mat srcnorm;
    normalize(src, srcnorm, 0, 255, NORM_MINMAX, CV_8UC1);
    return srcnorm;
}

int main(int argc, char *argv[]) {
    vector<Mat> db;

    // load greyscale images (these are from http://www.cl.cam.ac.uk/research/dtg/attarchive/facedatabase.html)
    db.push_back(imread("s1/1.pgm",0));
    db.push_back(imread("s1/2.pgm",0));
    db.push_back(imread("s1/3.pgm",0));

    db.push_back(imread("s2/1.pgm",0));
    db.push_back(imread("s2/2.pgm",0));
    db.push_back(imread("s2/3.pgm",0));

    db.push_back(imread("s3/1.pgm",0));
    db.push_back(imread("s3/2.pgm",0));
    db.push_back(imread("s3/3.pgm",0));

    db.push_back(imread("s4/1.pgm",0));
    db.push_back(imread("s4/2.pgm",0));
    db.push_back(imread("s4/3.pgm",0));

    int total = db[0].rows * db[0].cols;

    // build matrix (column)
    Mat mat(total, db.size(), CV_32FC1);
    for(int i = 0; i < db.size(); i++) {
        Mat X = mat.col(i);
        db[i].reshape(1, total).col(0).convertTo(X, CV_32FC1, 1/255.);
    }

    // Change to the number of principal components you want:
    int numPrincipalComponents = 12;

    // Do the PCA:
    PCA pca(mat, Mat(), CV_PCA_DATA_AS_COL, numPrincipalComponents);

    // Create the Windows:
    namedWindow("avg", 1);
    namedWindow("pc1", 1);
    namedWindow("pc2", 1);
    namedWindow("pc3", 1);

    // Mean face:
    imshow("avg", pca.mean.reshape(1, db[0].rows));

    // First three eigenfaces:
    imshow("pc1", normalize(pca.eigenvectors.row(0)).reshape(1, db[0].rows));
    imshow("pc2", normalize(pca.eigenvectors.row(1)).reshape(1, db[0].rows));
    imshow("pc3", normalize(pca.eigenvectors.row(2)).reshape(1, db[0].rows));

    // Show the windows:
    waitKey(0);
}

如果你想逐行構建矩陣(比如上面的原始問題),請改用:

// build matrix
Mat mat(db.size(), total, CV_32FC1);
for(int i = 0; i < db.size(); i++) {
    Mat X = mat.row(i);
    db[i].reshape(1, 1).row(0).convertTo(X, CV_32FC1, 1/255.);
}

並將PCA中的標志設置為:

CV_PCA_DATA_AS_ROW

關於機器學習。 我使用OpenCV C ++ API編寫了一個關於機器學習的文檔,其中包含大多數分類器的示例,包括支持向量機。 也許你可以在那里得到一些靈感: http//www.bytefish.de/pdf/machinelearning.pdf

data.row(i) = projectedMat.row(0);

這不行。 operator=是淺拷貝,意味着實際上沒有復制數據。 采用

cv::Mat sample = data.row(i); // also a shallow copy, points to old data!
projectedMat.row(0).copyTo(sample);

同樣也適用於:

desc_mat.row(i) = images[i].reshape(1, 1);

我建議在svn head中查看新檢查的測試

模塊/核心/測試/ test_mat.cpp

在線: https//code.ros.org/svn/opencv/trunk/opencv/modules/core/test/test_mat.cpp

在舊c和新c ++中有PCA的例子

希望有所幫助!

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM