PCA + SVM с использованием синтаксиса C ++ в OpenCV 2.2 - PullRequest
3 голосов
/ 12 февраля 2011

У меня проблемы с работой PCA и Eigenfaces с использованием новейшего синтаксиса C ++ с классами Mat и PCA. Более старый синтаксис C использовал массив IplImage * в качестве параметра для выполнения своей обработки, а текущий API принимает только Mat, отформатированный в Column или Row. Я использовал подход Row, используя функцию изменения формы, чтобы подогнать матрицу моего изображения под одну строку. В конце концов я хочу взять эти данные и затем использовать алгоритм 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");

Ответы [ 3 ]

9 голосов
/ 19 апреля 2011

То, что сказал этарион, верно.

Чтобы скопировать столбец или строку, вы всегда должны написать:

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

Следующая программа показывает, как выполнить PCA в OpenCV. Он покажет среднее изображение и первые три Eigenfaces. Изображения, которые я там использовал, доступны с 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

Относительно машинного обучения. Я написал документ по машинному обучению с помощью API OpenCV C ++, в котором есть примеры для большинства классификаторов, включая машины опорных векторов. Может быть, вы можете получить вдохновение там: http://www.bytefish.de/pdf/machinelearning.pdf.

4 голосов
/ 12 февраля 2011
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);
0 голосов
/ 12 февраля 2011

Я бы посоветовал посмотреть недавно проверенные тесты в svn head

modules / core / test / test_mat.cpp

онлайн здесь: https://code.ros.org/svn/opencv/trunk/opencv/modules/core/test/test_mat.cpp

примеры для PCA в старом c и новом c ++

Надеюсь, это поможет!

...