svm预测函数总是返回一个大数

时间:2015-11-25 16:38:33

标签: opencv svm predict

我试图实现一个简单的样本,目的是转换并检测一个数字。来源如下:

第1步:转换的初始数据: 我使用图像编号收集来处理数据,

    Mat vectorMatToMat(vector<Mat> list) {
        if (list.empty()) {
            return Mat();
        }
        int row = list.size();
        int col = list.at(0).rows * list.at(0).cols;
        Mat data(row, col, CV_32FC1);
        int i = 0;
        for (i = 0; i < row; i++) {
            Mat rowMat = list.at(i).reshape(1, 1);
            rowMat.copyTo(data.row(i));
        }
        return data;
    }
    Mat vectorIntToMat(vector<int> list) {
        int row = list.size();
        Mat data(row, 1, CV_32SC1, &list[0]);
        return data;
    }

    Mat dataMat = vectorMatToMat(listImageForTraining);
    Mat dataLabel = vectorIntToMat(listLabel);

第2步:初始化SVM:

Ptr<TrainData> trainData = TrainData::create(dataMat, ROW_SAMPLE, dataLabel);

Ptr<SVM> model = SVM::create();
model->setType(SVM::C_SVC);
model->setKernel(SVM::LINEAR);
model->setC(7);
model->setNu(SVM::NU_SVC);
model->setP(0);
model->setDegree(0);
model->setGamma(20);
model->setCoef0(0);
TermCriteria term(CV_TERMCRIT_ITER +CV_TERMCRIT_EPS, 1000, 1e-6);
model->setTermCriteria(term);
model->train(trainData);

第3步:尝试使用SVM进行预测:

    int i = 0;
    for (i = 0; i < 15; i++) {
        Mat check = dataMat.row(i);
        ostringstream oss;
        oss << i;
        imshow(oss.str(), check.reshape(1, 128));

        check = check.reshape(1, 1);
        int lable = svm.predict(check);

        cout << "Result: " << lable << endl;
    }

第4步:结果:

Result: -1237234688
Result: 159407376
Result: 159407376
Result: 167908848
Result: 1065353216
Result: 1065353216
Result: 1065353216
Result: 1065353216
Result: 1065353216
Result: 1065353216
Result: 1065353216
Result: 1065353216
Result: 1065353216
Result: 1065353216
Result: 1065353216

正如我们所看到的,我的结果是非常大的数字,虽然我的标签是0到10的数字。 我无法理解为什么,我认为在初始化SVM模型时我有一个错误。

但我不知道如何解决这个问题, 如果有任何想法,请帮助我。

谢谢&amp;最诚挚的问候,

Thiep

2 个答案:

答案 0 :(得分:0)

在以下方法的第一步看起来有误:

Mat vectorIntToMat(vector<int> list) {
    int row = list.size();
    Mat data(row, 1, CV_32SC1, &list[0]);
    return data;
}

按值传递标签矢量。它实际上制作了一个标签的矢量副本,它将在返回后被破坏。 比你使用cv::Mat的以下构造函数:

Mat::Mat(int rows, int cols, int type, void* data, size_t step=AUTO_STEP)

根据OpenCV documentation,此构造函数不分配矩阵数据。它只使用你提供的内存。但是在你的情况下,这个内存将在方法返回后被释放,而标签的矩阵对象将是不正确的。

显然,如果您在训练中使用不正确的标签矩阵,则会得到不正确的预测结果。

我认为,问题的最简单解决方法如下:

Mat vectorIntToMat(vector<int> list) {
    int row = list.size();
    Mat data = Mat(row, 1, CV_32SC1, &list[0]).clone();
    return data;
}

希望它会有所帮助。

答案 1 :(得分:0)

我尝试修改如下,但结果仍然不好。

Mat vectorIntToMat(vector<int> list) {
    int row = list.size();
    Mat data = Mat(row, 1, CV_32SC1, &list[0]).clone();
    return data;
}

之后我尝试修改如下,我的问题解决了:

Mat vectorIntToMat(vector<int> list) {
    return Mat(list, true);
}

但我仍然不明白根本原因。 有谁可以帮我解释一下?