我正在尝试使用SVM、openCV、C++和Visual Studio 2008(MFC应用程序)创建一个简单的OCR应用程序。
我的训练样本是机器打印的数字(0-9)的二进制图像。我想使用DAGSVM来解决这个多类问题。因此,我需要创建45个SVM,每个SVM都是两个类别的SVM(SVM(0,1), SVM(0,2)… SVM(8,9))。
事情进展如下:
SVM的参数:
CvSVMParams params;params.svm_type = CvSVM::C_SVC;params.kernel_type = CvSVM::LINEAR;params.term_crit = cvTermCriteria(CV_TERMCRIT_ITER, 100, 1e-6);
类别i的训练图像数据存储在矩阵trainData[i]中(每行是一个28×28图像的像素,这意味着矩阵有784列)。在训练每个SVM时,我创建了两个名为curTrainData和curTrainLabel的矩阵。
for (int i = 0; i < 9; i++) for (int j = i+1; j < 10; j++) { curTrainData.create(trainData[i].rows + trainData[j].rows, 784, CV_32FC1); curTrainLabel.create(curTrainData.rows, 1, CV_32FC1); // 合并两个矩阵:trainData[i] 和 trainData[j] for (int k = 0; k < trainData[i].rows; k++) { curTrainLabel.at<float>(k, 0) = 1.0; // 数字i的类别 for (int l = 0; l < 784; l++) curTrainData.at<float>(k,l) = trainData[i].at<float>(k,l); } for (int k = 0; k < trainData[j].rows; k++) { curTrainLabel.at<float>(k + trainData[i].rows, 0) = -1.0; // 数字j的类别 for (int l = 0; l < 784; l++) curTrainData.at<float>(k + trainData[i].rows,l) = trainData[j].at<float>(k,l); } svms[i][j].train(curTrainData, curTrainLabel, Mat(), Mat(), params); }
我在调用svms[i][j].train…时遇到了错误。完整的错误信息是:
Unhandled exception at 0x75b5d36f in svm.exe: Microsoft C++ exception: cv::Exception at memory location 0x0022af8c..
说实话,我对openCV中实现的SVM并不完全理解,我也找不到任何关于它们如何处理图像中对象的示例。
如果有人能告诉我哪里出了问题,我将不胜感激 🙁
更新 09/03:我弄错了。错误来自于:
str.Format(_T("Results\trained_%d_%d.xml"), i, j); svms[i][j].save(CT2A(str));
str是一个CString变量。
即使我改成如下代码,问题仍然存在:
svms[i][j].save("Results\trained.xml");
我已经创建了Results文件夹,其他文件(使用fopen()、imwrite()等方法的文件)都能很好地写入其中。我不知道为什么在使用SVM的save方法时无法添加文件夹。
回答:
如果你使用反斜杠”\”,你必须使用”\\”代替(或者你可以使用正斜杠”/”)。