HOG特征+SVM训练过程
(2018-08-05 20:30:13)分类: 机器视觉 |
转:https://blog.csdn.net/yangleo1987/article/details/53185580
言归正传,实验所用行人库:INRIAPerson。训练过程如下:
1、选定正负样本:正样本、行人库中正样本大小为96x160,比我们需要的64x128窗口要大,是因为每一个边有一个padding,16像素,因此保持中心不动,选取64x128窗口大小的图片作为训练的正样本。负样本、负样本大小不统一,在每一个负样本图像中随机选取10个64x128大小的patch作为训练的负样本(即用来训练的负样本个数是行人库中个数的10倍,每一个负样本产生10个patch);
2、提取hog特征(opencv完成);
3、将提取出来的hog特征投入到svm中训练,得到初始分类器;
4、利用初始分类器(其实就是支持向量以及对应的权值,还有一个偏移)的支持向量和对应的权值加权得到检测行人的检测子,再加上一维的偏移,整个检测子是3781维。但是,重点来了,这个初始检测子,效果很差很差,不信你可以试试,有很多错检的(有点hard
5、利用这个初始的检测子去检测前面用来训练的负样本原图(不是随机提取出来的图,而是原图),检测方法是利用cvhog的多尺度检测detectMultiScale方法,检测出来一打一打的行人(但实际上是负样本,所以肯定是错的),这些检测出来的区域,就是hard
6、提取第5步中hard
7、将hard
之前纠结过的地方:
1、负样本为何会大小不统一?因为整个负样本中是没有人的,随便在图像中选择64x128窗口大小的图像都可作为负样本进行训练,因此负样本原始图像的大小并不重要。
2、resize不能乱用。这里就迁出了为何要使用多尺度检测的问题。因为我们规定的窗口大小是64x128,太大或者太小的人都检测不到,因此我们利用尺度对图像进行缩放,在每一层中进行检测,最后在原图画矩形框,他的结果是把每一层的检测结果综合起来,可能有一层就检测不到了,resize的话没准就恰恰resize成了找不到行人或者把不是行人的判断为行人了。resize不能乱用,我最开始用resize的时候,支持向量和投入训练的数量一样多,这不坑爹么?所以不到万不得已,别用resize。
3、hard
重新训练行人检测的流程:
(1)准备训练样本集合;包括正样本集和负样本集;根据机器学习的基础知识我们知道,要利用机器学习算法进行样本训练,从而得到一个性能优良的分类器,训练样本应该是无限多的,而且训练样本应该覆盖实际应用过程中可能发生的各种情况。(很多朋友,用10来个正样本,10来个负样本进行训练,之后,就进行测试,发现效果没有想象中的那么好,就开始发牢骚,抱怨。。。对于这些人,我只能抱歉的说,对于机器学习、模式识别的认识,你还处于没有入门的阶段);实际应用过程中,训练样本不可能无限多,但无论如何,三五千个正样本,三五千个负样本,应该不是什么难事吧?(如果连这个都做不到,建议你别搞机器学习,模式识别了;训练素材都没有,怎么让机器学习到足够的信息呢?)
(2)收集到足够的训练样本之后,你需要手动裁剪样本。例如,你想用Hog+SVM来对商业步行街的监控画面中进行行人检测,那么,你就应该用收集到的训练样本集合,手动裁剪画面中的行人(可以写个简单程序,只需要鼠标框选一下,就将框选区域保存下来)。
(3)裁剪得到训练样本之后,将所有正样本放在一个文件夹中;将所有负样本放在另一个文件夹中;并将所有训练样本缩放到同样的尺寸大小。OpenCV自带的例子在训练时,就是将样本缩放为64*128进行训练的;
(4)提取所有正样本的Hog特征;
(5)提取所有负样本的Hog特征;
(6)对所有正负样本赋予样本标签;例如,所有正样本标记为1,所有负样本标记为0;
(7)将正负样本的Hog特征,正负样本的标签,都输入到SVM中进行训练;Dalal在论文中考虑到速度问题,建议采用线性SVM进行训练。这里,不妨也采用线性SVM;
(8)SVM训练之后,将结果保存为文本文件。
(9)线性SVM进行训练之后得到的文本文件里面,有一个数组,叫做support
下面给出样本训练的参考代码:
-
class Mysvm: public CvSVM
-
{
-
public:
-
int get_alpha_count()
-
{
-
return this->sv_total;
-
}
-
-
int get_sv_dim()
-
{
-
return this->var_all;
-
}
-
-
int get_sv_count()
-
{
-
return this->decision_func->sv_count;
-
}
-
-
double* get_alpha()
-
{
-
return this->decision_func->alpha;
-
}
-
-
float** get_sv()
-
{
-
return this->sv;
-
}
-
-
float get_rho()
-
{
-
return this->decision_func->rho;
-
}
-
};
-
-
void Train()
-
{
-
char classifierSavePath[256] = "c:/pedestrianDetect-peopleFlow.txt";
-
-
string positivePath = "E:\\pictures\\train1\\pos\\";
-
string negativePath = "E:\\pictures\\train1\\neg\\";
-
-
int positiveSampleCount = 4900;
-
int negativeSampleCount = 6192;
-
int totalSampleCount = positiveSampleCount + negativeSampleCount;
-
-
cout<<"//////////////////////////////////////////////////////////////////"< featureVec;
-
-
hog.compute(img, featureVec, cv::Size(8,8));
-
int featureVecSize = featureVec.size();
-
-
for (int j=0; jdata.fl[i] = 1;
-
}
-
cout<<"end of training for positive samples..."< featureVec;
-
-
hog.compute(img,featureVec,cv::Size(8,8));//计算HOG特征
-
int featureVecSize = featureVec.size();
-
-
for ( int j=0; jdata.fl[ i + positiveSampleCount ] = -1;
-
}
-
-
cout<<"end of training for negative samples..."1764), svm.get_support_vector(i), 1764*sizeof(float));
-
}
-
-
double* alphaArr = svm.get_alpha();
-
int alphaCount = svm.get_alpha_count();
-
-
for(int i=0; idata.fl[i] = alphaArr[i];
-
}
-
cvMatMul(alp, sv, re);
-
-
int posCount = 0;
-
for (int i=0; i<<span class="hljs-number">1764; i++)
-
{
-
re->data.fl[i] *= -1;
-
}
-
-
FILE* fp = fopen("c:/hogSVMDetector-peopleFlow.txt","wb");
-
if( NULL == fp )
-
{
-
return 1;
-
}
-
for(int i=0; i<<span class="hljs-number">1764; i++)
-
{
-
fprintf(fp,"%f \n",re->data.fl[i]);
-
}
-
float rho = svm.get_rho();
-
fprintf(fp, "%f", rho);
-
cout<<"c:/hogSVMDetector.txt 保存完毕"<</div>
接着,再给出利用训练好的分类器进行行人检测的参考代码:
-
void Detect()
-
{
-
CvCapture* cap = cvCreateFileCapture("E:\\02.avi");
-
if (!cap)
-
{
-
cout<<"avi file load error..."< x;
-
ifstream fileIn("c:/hogSVMDetector-peopleFlow.txt", ios::in);
-
float val = 0.0f;
-
while(!fileIn.eof())
-
{
-
fileIn>>val;
-
x.push_back(val);
-
}
-
fileIn.close();
-
-
vector found;
-
cv::HOGDescriptor hog(cv::Size(64,64), cv::Size(16,16), cv::Size(8,8), cv::Size(8,8), 9);
-
hog.setSVMDetector(x);
-
-
IplImage* img = NULL;
-
cvNamedWindow("img", 0);
-
while(img=cvQueryFrame(cap))
-
{
-
hog.detectMultiScale(img, found, 0, cv::Size(8,8), cv::Size(32,32), 1.05, 2);
-
if (found.size() > 0)
-
{
-
-
for (int i=0; i