opencv 人脸识别 (二)训练和识别

opencv 人脸识别 (二)训练和识别上一篇中我们对训练数据做了一些预处理,检测出人脸并保存在\pic\color\x文件夹下(x=1,2,3,…类别号),本文做训练和识别。为了识别,首先将人脸训练数据转为灰度、对齐、归一化,再放入分类器(EigenFaceRecognizer),最后用训练出的model进行predict。—————————————–环境:vs2010+op

大家好,又见面了,我是你们的朋友全栈君。

上一篇中我们对训练数据做了一些预处理,检测出人脸并保存在\pic\color\x文件夹下(x=1,2,3,…类别号),本文做训练和识别。为了识别,首先将人脸训练数据 转为灰度、对齐、归一化,再放入分类器(EigenFaceRecognizer),最后用训练出的model进行predict。


—————————————–


环境:vs2010+opencv 2.4.6.0

特征:eigenface

Input:一个人脸数据库,15个人,每人20个样本(左右)。

Output:人脸检测,并识别出每张检测到的人脸。


—————————————–


1. 为训练数据预处理( 转为灰度、对齐、归一化 

  • 转为灰度和对齐是后面做训练时EigenFaceRecognizer的要求;
  • 归一化是防止光照带来的影响

在上一篇的 2.2 Prehelper.cpp文件中加入函数

void resizeandtogray(char* dir,int k, vector<Mat> &images, vector<int> &labels,
vector<Mat> &testimages, vector<int> &testlabels);

void resizeandtogray(char* dir,int K, vector<Mat> &images, vector<int> &labels,	vector<Mat> &testimages, vector<int> &testlabels){	IplImage* standard = cvLoadImage("D:\\privacy\\picture\\photo\\2.jpg",CV_LOAD_IMAGE_GRAYSCALE);	string cur_dir;	char id[5];	int i,j;	for(int i=1; i<=K; i++)	{		cur_dir = dir;		cur_dir.append("gray\\");			_itoa(i,id,10);		cur_dir.append(id);		const char* dd = cur_dir.c_str();		CStatDir statdir;		if (!statdir.SetInitDir(dd))		{			puts("Dir not exist");			return;		}		cout<<"Processing samples in Class "<<i<<endl;		vector<char*>file_vec = statdir.BeginBrowseFilenames("*.*");		for (j=0;j<file_vec.size();j++)		{			IplImage* cur_img = cvLoadImage(file_vec[j],CV_LOAD_IMAGE_GRAYSCALE);			cvResize(cur_img,standard,CV_INTER_AREA);			Mat cur_mat = cvarrToMat(standard,true),des_mat;			cv::normalize(cur_mat,des_mat,0, 255, NORM_MINMAX, CV_8UC1);			cvSaveImage(file_vec[j],cvCloneImage(&(IplImage) des_mat));			if(j!=file_vec.size())			{					images.push_back(des_mat);					labels.push_back(i);			}			else			{				testimages.push_back(des_mat);				testlabels.push_back(i);			}		}		cout<<file_vec.size()<<" images."<<endl;	}}


并在main中调用:

int main( )
{
	CvCapture* capture = 0;
	Mat frame, frameCopy, image;
	string inputName;	
	int mode;

	char dir[256] = "D:\\Courses\\CV\\Face_recognition\\pic\\"; 
	//preprocess_trainingdata(dir,K); //face_detection and extract to file
	vector<Mat> images,testimages;
	vector<int> labels,testlabels;
	resizeandtogray(dir,K,images,labels,testimages,testlabels); //togray, normalize and resize
	
	system("pause");
	return 0;
}

2. 训练

有了vector<Mat> images,testimages; vector<int> labels,testlabels; 可以开始训练了,我们采用EigenFaceRecognizer建模。

在Prehelper.cpp中加入函数

Ptr<FaceRecognizer> Recognition(vector<Mat> images, vector<int> labels,vector<Mat> testimages, vector<int> testlabels);

Ptr<FaceRecognizer> Recognition(vector<Mat> images, vector<int> labels,	vector<Mat> testimages, vector<int> testlabels){	Ptr<FaceRecognizer> model = createEigenFaceRecognizer(10);//10 Principal components	cout<<"train"<<endl;	model->train(images,labels);	int i,acc=0,predict_l;	for (i=0;i<testimages.size();i++)	{		predict_l = model->predict(testimages[i]);		if(predict_l != testlabels[i])		{			cout<<"An error in recognition: sample "<<i+1<<", predict "<<				predict_l<<", groundtruth "<<testlabels[i]<<endl;			imshow("error 1",testimages[i]);			waitKey();		}		else			acc++;	}	cout<<"Recognition Rate: "<<acc*1.0/testimages.size()<<endl;	return model;}


Recognization()输出分错的样本和正确率,最后返回建模结果Ptr<FaceRecognizer> model


主函数改为:

int main( )
{
	CvCapture* capture = 0;
	Mat frame, frameCopy, image;
	string inputName;	
	int mode;

	char dir[256] = "D:\\Courses\\CV\\Face_recognition\\pic\\"; 
	//preprocess_trainingdata(dir,K); //face_detection and extract to file
	vector<Mat> images,testimages;
	vector<int> labels,testlabels;
	//togray, normalize and resize; load to images,labels,testimages,testlabels
	resizeandtogray(dir,K,images,labels,testimages,testlabels); 
	//recognition
	Ptr<FaceRecognizer> model = Recognition(images,labels,testimages,testlabels);
	char* dirmodel = new char [256];
	strcpy(dirmodel,dir); strcat(dirmodel,"model.out");
	FILE* f = fopen(dirmodel,"w");
	fwrite(model,sizeof(model),1,f);
	system("pause");
	return 0;
}


最终结果:一个错分样本,正确率93.3%

opencv 人脸识别 (二)训练和识别

opencv 人脸识别 (二)训练和识别



文章所用代码打包链接:http://download.csdn.net/detail/abcjennifer/7047853



关于Computer Vision更多的学习资料将继续更新,敬请关注本博客和新浪微博Rachel Zhang



版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/137542.html原文链接:https://javaforall.net

(0)
上一篇 2022年6月1日 下午8:16
下一篇 2022年6月1日 下午8:16


相关推荐

  • java map遍历的几种方式_二叉树的三种遍历方式

    java map遍历的几种方式_二叉树的三种遍历方式一:在for循环中使用entries实现Map的遍历:二:在for循环中遍历key或者values,一般适用于只需要map中的key或者value时使用,在性能上比使用entrySet较好;三:通过Iterator遍历;

    2026年2月8日
    5
  • linux 实时查看日志 最新最后100行 tail「建议收藏」

    linux 实时查看日志 最新最后100行 tail「建议收藏」(1)实时查看日志文件tail-f日志文件名(2)只查看日志文件后100行tail-f-n100日志文件名(3)搜寻字符串grep‘搜寻字符串’日志文件名按ctrl+c退出————————————————版权声明:本文为CSDN博主「wanghai76」的原创文章,遵循CC4.0BY-SA版权协议,转载请附上原文出处链接及本声明。原文链接:http…

    2022年6月4日
    1.6K
  • Android原生编解码接口 MediaCodec 之——踩坑

    Android原生编解码接口 MediaCodec 之——踩坑关键帧MediaCodec有两种方式触发输出关键帧,一是由配置时设置的KEY_FRAME_RATE和KEY_I_FRAME_INTERVAL参数自动触发,二是运行过程中通过setParameters手动触发输出关键帧。自动触发输出关键帧在MediaCodec硬编码中设置I(关键帧)时间间隔,在api中是这么设置的mediaFormat.setInteger(MediaF………

    2025年8月7日
    6
  • Centos7配置MySQL读写分离[通俗易懂]

    Centos7配置MySQL读写分离[通俗易懂]背景82和83两台Mysql已经实现了主从复制。即将使用mysql-proxy进行读写分离,配置192为proxy,写82读83。mysql-proxymysql-proxy是一个处于客户端和服务器之间的中间件,它可以检测,分析或改变它们的通信。简单说就是一个中间层代理,一个连接池,负责将前台应用的连接请求转发给后台的数据库,并且通过lua脚本,可以实现复杂的连接控制和过滤,从而实现读写分离和负载…

    2022年5月30日
    54
  • 5G NSA 和SA 介绍

    5G NSA 和SA 介绍5G 组网模式 NSA 和 SANSA Non Standalone 非独立组网 SA Standalone 独立组网在 5G 网络架构中 3GPPTSG RAN 第 72 次全体大会上 提出了 8 个选项其中 选项 1 选项 2 选项 5 选项 6 是独立组网选项 3 选项 4 选项 7 和选项 8 是非独立组网 3 4 7 还有不同的子选项 5G 独立组网 选项 1 4G 核心网 4G 基站 UE

    2026年3月26日
    2
  • 剥析surging的架构思想

    剥析surging的架构思想1 前言 nbsp 前面第一篇阐述了采用基于 NETCORE 微服务架构 应用 surging 服务端与客户端之间进行通信的简单示例以及对于 surging 服务化框架简单介绍 在这篇文章中 我们将剥析 surging 的架构思想 surging 源码下载 2 通信机制 2 1 简介 nbsp nbsp nbsp nbsp 在单体应用中 模块之间的调用通信通过引用加载方法或者函数来实现 但是单体应用最终都会因为团队壮大 项目模块的扩展和

    2026年3月17日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号