目录

前言

准备工作

硬件

软件

训练过程

第一步:准备样本

第二步:生成样本描述文件

第三步:生成样本文件vec

第四步:训练分类器

过程中遇到的问题

总结


前言

目前主流的较前卫的目标检测方案是基于深度学习,而传统的方案则是基于手工特征,即通过对图形进行滑动窗口遍历计算机特征值,并训练特征分类器以达到检测的目的。本文则是基于级联分类器的样本训练过程的记录。

准备工作

硬件

可以长时间训练特征文件的电脑,样本文件的高宽比较大的时候,训练可能会耗费几十个小时甚至几天时间。

软件

  • 样本处理工具:推荐 XnConvert-win-x64.exe,可以批量重命名、旋转、镜像、灰度、缩放等。曾用过imagetuner_8.0.exe、reaConverterLite-Setup.exe,必要的时候还可以结合使用everything.exe重命名。imagetuner_8.0.exe功能比较简单,reaConverterLite-Setup.exe这个软件界面好用但是批量处理经常崩溃,而且需要通过资源管理器关闭界面再次开始,好处是重命名的功能,可以修改替换原有文件名,这个软件有收费版,没有尝试。

训练过程

第一步:准备样本

正样本与负样本。考虑各种场景下的正样本照片,负样本照片即为背景照片,正负样本的数量比以1:3为宜,如果负样本的图幅度很大,可以考虑降低负样本的数量,因为是通过正样本在负样本上的滑动遍历。

这个部分最大量的工作就是样本查找,对于正样本就是对其轮廓按照矩形或者圆形抠图,对与负样本就是寻找大量的背景图片。当样本数量不够时,可以考虑扩充样本。

正样本抠图

注意,不要用截屏工具。

扩充样本

对称、旋转

第二步:生成样本描述文件

可以通过cmd命令行实现,当然最方便可以写个程序一次生成,这是我用c++编写的两个小程序,目前用着没发现bug,够用。

正样本描述文件生成c++伪码

#include <iostream>
 #include <string>
 #include <io.h>
 #include <windows.h>
 #include <gdiplus.h>
 #pragma comment(lib, "gdiplus.lib")using namespace std;
 using namespace Gdiplus;int main()
 {
     //输入文件目录
     char dir[256] = { 0 };
     cout << "Enter a directory: ";
     cin.getline(dir, 255);
         //检查文件目录路径是否合法,添加"\\*.*"
     char dirNew[261] = { 0 };
     strcpy_s(dirNew, dir);
     strcat_s(dirNew, "\\*.*");
    //遍历目录下的文件,并读取图片信息,写入描述文件
     GdiplusStartupInput gdiplusstartupinput;
     ULONG_PTR gdiplustoken;
     GdiplusStartup(&gdiplustoken, &gdiplusstartupinput, NULL);    intptr_t handle;
     _finddata_t findData;    handle = _findfirst(dirNew, &findData);
     if (handle == -1)        // 检查是否成功
     {
         cout << "no file" << endl;
         system("Pause");
         return 0;
     }
     //创建空白描述文件
     char filePath[256] = { 0 };
     strcpy_s(filePath, dir);
     strcat_s(filePath, "\\posdata.dat");
     FILE *fp = NULL;
     errno_t err = fopen_s(&fp, filePath, "w+");
     if (err != 0)
     {
         cout << "open err" << endl;
         system("Pause");
         return 0;
     }    do
     {        //如果当前文件有子文件夹
         if (findData.attrib & _A_SUBDIR)
         {
             /*if (strcmp(findData.name, ".") == 0 || strcmp(findData.name, "..") == 0)
                 continue;*/            //cout << findData.name << "\t<dir>\n";
             在目录后面加上"\\"和搜索到的目录名进行下一次搜索
             //strcpy_s(dirNew, dir);
             //strcat_s(dirNew, "\\");
             //strcat_s(dirNew, findData.name);            //listFiles(dirNew);
         }
         else
         {
             cout << findData.name << "\t" << findData.size << " bytes.\n";
             //判断如果文件类型不是图片,则跳过
             string fileName(findData.name);
             string fileType = fileName.substr(strlen(findData.name) - 4);
             if (!(fileType == ".jpg" || fileType == ".jpeg" || fileType == ".jpe" || fileType == ".bmp"))
                 continue;            //添加文件夹名
             //fputs("cw45/", fp);            fputs(findData.name, fp);
             fputs(" 1 0 0 ", fp);            //路径字符转宽字符
             char ImgPath[512] = { 0 };
             sprintf_s(ImgPath, "%s%s%s", dir, "\\", findData.name);
             int len = 0;
             len = strlen(ImgPath);
             int unicodeLen = ::MultiByteToWideChar(CP_ACP, 0, ImgPath, -1, NULL, 0);
             wchar_t *pUnicode = new wchar_t[unicodeLen + 1];
             memset(pUnicode, 0, (unicodeLen + 1) * sizeof(wchar_t));
             ::MultiByteToWideChar(CP_ACP, 0, ImgPath, -1, (LPWSTR)pUnicode, unicodeLen);            wstring infilename((wchar_t*)pUnicode);
            Bitmap* bmp = new Bitmap(infilename.c_str());
             UINT height = bmp->GetHeight();
             UINT width = bmp->GetWidth();
             cout << "width " << width << ", height " << height << endl;            fprintf(fp, "%d", width);
            fputs(" ", fp);
            fprintf(fp, "%d", height);
            delete[] pUnicode;
             delete bmp;
             fputs("\n", fp);
         }
     } while (_findnext(handle, &findData) == 0);    // 关闭搜索句柄
    _findclose(handle);    
    GdiplusShutdown(gdiplustoken);
    fclose(fp);
    system("Pause");
    return 0;
 }

负样本描述文件生成C++伪码

#include <iostream>
 #include <string>
 #include <io.h>using namespace std;
 int main()
 {
     //输入文件目录
     char dir[256] = { 0 };
     cout << "Enter a directory: ";
     cin.getline(dir, 255);
         //TODO:检查文件目录路径是否合法,是否添加"\\*.*"
     char dirNew[261] = { 0 };
     strcpy_s(dirNew, dir);
     strcat_s(dirNew, "\\*.*");     intptr_t handle;
     _finddata_t findData;    handle = _findfirst(dirNew, &findData);
     if (handle == -1)        // 检查是否成功
     {
         cout << "no file" << endl;
         system("Pause");
         return 0;
     }
     //创建空白描述文件
     char filePath[256] = { 0 };
     strcpy_s(filePath, dir);
     strcat_s(filePath, "\\negdata.dat");
     FILE *fp = NULL;
     errno_t err = fopen_s(&fp, filePath, "w+");
     if (err != 0)
     {
         cout << "open err" << endl;
         system("Pause");
         return 0;
     }    do
     {
         if (findData.attrib & _A_SUBDIR)
         {
             //if (strcmp(findData.name, ".") == 0 || strcmp(findData.name, "..") == 0)
             //    continue;            //cout << findData.name << "\t<dir>\n";
             在目录后面加上"\\"和搜索到的目录名进行下一次搜索
             //strcpy_s(dirNew, dir);
             //strcat_s(dirNew, "\\");
             //strcat_s(dirNew, findData.name);            //listFiles(dirNew);
         }
         else
         {
             //cout << findData.name << "\n";
             //判断如果文件类型不是图片,则跳过
             string fileName(findData.name);
             string fileType = fileName.substr(strlen(findData.name) - 4);
             if (!(fileType == ".jpg" || fileType == ".jpeg" || fileType == ".jpe" || fileType == ".bmp"))
                 continue;            //路径字符转宽字符
             char ImgPath[512] = { 0 };
             sprintf_s(ImgPath, "%s%s%s", dir, "\\", findData.name);            cout << ImgPath << "\n";
            fputs(ImgPath, fp);
            fputs("\n", fp);
         }
     } while (_findnext(handle, &findData) == 0);    _findclose(handle);    // 关闭搜索句柄
    fclose(fp);
    system("Pause");
    return 0;
 }

第三步:生成样本文件vec

第四步:训练分类器

打开windows命令处理程序,红色字符输入,黑色为输出

Microsoft Windows [版本 6.1.7601]

版权所有 (c) 2009 Microsoft Corporation。保留所有权利。

C:\Users\admin>cd /d G:\

G:\>cd JLCZ\OpenCV\opencv-3.4.2-vc14_vc15\opencv\build\x64\vc14\bin\

G:\JLCZ\OpenCV\opencv-3.4.2-vc14_vc15\opencv\build\x64\vc14\bin>

opencv_createsamples.exe -info G:\JLCZ\MachineVision\03_CreateSamples\v0.2\posdata\posdata.dat -vec G:\JLCZ\MachineVision\03_CreateSamples\v0.2\airplane_wheel_0_2.vec -num 5206 -w 150 -h 150

Info file name: G:\JLCZ\MachineVision\03_CreateSamples\v0.2\posdata\posdata.dat

Img file name: (NULL)

Vec file name: G:\JLCZ\MachineVision\03_CreateSamples\v0.2\airplane_wheel_0_2.vec

BG  file name: (NULL)

Num: 5206

BG color: 0

BG threshold: 80

Invert: FALSE

Max intensity deviation: 40

Max x angle: 1.1

Max y angle: 1.1

Max z angle: 0.5

Show samples: FALSE

Width: 150

Height: 150

Max Scale: -1

RNG Seed: 12345

Create training samples from images collection...

Done. Created 5206 samples

G:\JLCZ\OpenCV\opencv-3.4.2-vc14_vc15\opencv\build\x64\vc14\bin>

opencv_traincascade.exe -data G:\JLCZ\MachineVision\04_CreateClassifier\v0.2 -vec G:\JLCZ\MachineVision\03_CreateSamples\v0.2\airplane_wheel_0_2.vec -bg G:\JLCZ\MachineVision\03_CreateSamples\v0.2\negdata\negdata.dat -numPos 4900 -numNeg 4284 -numStages 15 -featureType LBP -w 150 -h 150 -minHitRate 0.995 -maxFalseAlarmRate 0.5

过程中遇到的问题

1.vs动态链接库指针报错,提示为空

原因:参数错误,参数与值之间没有空格,添加空格就好了。

javacv 分类器训练 opencv训练分类器需要多久_ico

2.cmd训练出现“Train dataset for temp stage can not be filled. Branch training terminated ..."

解决:负样本图片路径错误,在描述文件中*.dat中修改就好了。

3.在createsamples执行之后需要注意cmd返回的命令,生成了多少样本,以该样本数量为准。假设你有5206和样本,但是createsamples只创建了1000个,意味着vec文件中只存储了1000个,这是有问题的,参数应该与返回值相同。检查参数发现 正样本数量的参数应该是“-num”,而输入为“_num”则出现了这种情况。解决办法为重新确认参数,生成vec。

4.opencv_traincascade.exe执行的参数,输入的正样本数量numPos需要小于实际得到的正样本数量,numPos+(numStages-1)*numPos*(1-minHitRate)《=准备的训练样本

5.

javacv 分类器训练 opencv训练分类器需要多久_javacv 分类器训练_02

6.在控制台使用system(“pause”)不显示图像

为什么opencv里面这个用system(“pause”)就不能载入图像?

总结

(这本是去年未完成的部分,可惜一直拖到现在,以至于想不起来具体总结的内容,下次使用这个训练器之后再来总结吧)