C++开发人脸性别识别教程(10)——加入图片的人脸检測程序
现在我们的MFC框架已经初具规模,能够读取并显示目录下的图片。在这篇博文中我们将向当中加入人脸检測的程序。
一、人脸检測算法
这里我们使用OpenCv封装的Adaboost方法来进行人脸检測,參见:C++开发人脸性别识别教程(4)——OpenCv的人脸检測函数
二、初始化
1、加入初始化button
在进行人脸检測之前须要初始化一些相关变量。比如开辟内存,载入检測器等等。首先,我们为MFC框架加入一个初始化button。并将ID更改为IDC_BUTTON_INITIAL:

双击这个button。加入事件响应函数:

2、初始化变量
从之前的博客中可知。OpenCv在进行人脸检測时须要用到两个静态变量:static CvMemStorage* storage和static CvHaarClassifierCascade* cascade。这里我们将其作为成员变量加入到CGenderRecognitionMFCDlg类中。这里因为static CvMemStorage*和static CvHaarClassifierCascade*这两个类型名在MFC类向导中是无法被识别的,因此须要手动加入至CGenderRecognitionMFCDlg类的构造函数中:

接下里对这两个惊天变量进行初始化。
C++明白规定静态成员变量要在类外进行初始化,而不能在类内声明时或者构造函数内进行初始化,原因就是静态变量时属于类本身的,而非类对象的属性,和全局变量相似,因此我们将这两个静态变量的初始化操作放在GenderRecognitionMFCDlg.cpp文件(从解决方式资源管理器窗体中查找cpp文件)的开头位置:
// 用于应用程序“关于”菜单项的 CAboutDlg 对话框
CvMemStorage* CGenderRecognitionMFCDlg::storage = NULL;
CvHaarClassifierCascade* CGenderRecognitionMFCDlg::cascade = NULL;
然后在“初始化”button的响应函数OnBnClickedButtonInitial()中载入相应的人脸检測器:
void CGenderRecognitionMFCDlg::OnBnClickedButtonInitial()
{
cascade = cvLoadHaarClassifierCascade("D:\\opencv\\sources\\data\\haarcascades
\\haarcascade_frontalface_alt_tree.xml",cvSize(30,30));
storage = cvCreateMemStorage(0);
// TODO: 在此加入控件通知处理程序代码
}
初始化完毕。
三、编写人脸检測函数
这里将人脸检測的操作封装成一个函数detect_and_draw(),作为成员函数加入到CGenderRecognitionMFCDlg类中:

在类视图中找到detect_and_draw()函数。完好其人脸检測代码,因为之前已经具体介绍过人脸检測的相关操作。这里直接给出代码:
void CGenderRecognitionMFCDlg::detect_and_draw(IplImage* img)
{
/**********初始化**********/
double scale = 1.2;
IplImage* gray = cvCreateImage(cvSize(img->width,img->height),8,1); /**********灰度化**********/
if (img->nChannels = 3)
{
cvCvtColor(img,gray, CV_BGR2GRAY);//将图像灰度化存放在gray中
}
else
{
gray = img;
} /**********直方图均衡**********/
cvEqualizeHist(gray,gray); /**********人脸检測**********/
cvClearMemStorage(storage);
CvSeq* objects = cvHaarDetectObjects(gray,//待检測图像
cascade, //分类器标识
storage, //存储检測到的候选矩形
1.3, //相邻两次检測中窗体扩大的比例
3, //觉得是人脸的最小矩形数(阈值)
0, //CV_HAAR_DO_CANNY_PRUNING
cvSize(30,30)); //初始检測窗体大小 /**********绘制检測结果**********/
if(objects->total > 0) //假设人脸检測成功
{
for (int i = 0; i < (objects ? objects->total : 0); i++)
{
CvRect* rect = (CvRect*)cvGetSeqElem(objects,i);
cvRectangle(img,cvPoint(rect->x,rect->y),
cvPoint(rect->x + rect->width,rect->y + rect->height),cvScalar(0.0,255));
}
} /**********在图像控件上显示图像**********/
CvvImage cvvImage;
cvvImage.CopyOf(img);
cvvImage.DrawToHDC(m_pPicCtlHdc,m_PicCtlRect); cvReleaseImage(&gray);
}
注意这里相对于之前的程序,加入了一项直方图均衡化的操作。以提高人脸检測的成功率:

四、调用人脸检測函数
理论上在显示图像之前应该自己主动调用人脸检測操作,因此在GetNextBigImg()函数中调用人脸检測函数:

因为在detect_and_draw()函数中已经封装了picture显示的程序。所以能够将GetNextBigImg()函数中原有的picture控件显示程序去掉。
大功告成。顺利完毕人脸检測:

三、总结
这里我们初步完毕了MFC中的人脸检測功能。但这里存在两个严重的BUG,一是假设用户未单击“初始化”button,直接打开图片,程序会因缺少必要的初始化步骤而直接崩溃;二是如上图所见,OpenCv在进行人脸检測时可能会错误检測出多个矩形。当中仅仅有一个矩形包括人脸,其余的都是干扰,须要进行处理,我们将在下一篇博客中介绍怎样解决这两个BUG。
同一时候在此须要强调一下两个问题:
1、静态成员变量的初始化:c++中能够对类中私有成员中的静态变量初始化吗?
2、字符串的连接:C++字符换行 .
C++开发人脸性别识别教程(10)——加入图片的人脸检測程序的更多相关文章
- C++开发人脸性别识别教程(16)——视频人脸性别识别
在之前的博文中我们已经可以顺利驱动摄像头来採集源图像.在这篇博文中将正式为其加入性别识别的代码,实现摄像头视频的人脸性别识别. 一.人脸检測 在得到摄像头採集的源图像之后,首先要做的就是对其进行人脸检 ...
- C++开发人脸性别识别教程(19)——界面美化
在这篇博文中将完毕<C++开发人脸性别识别>的收尾工作.主要内容分为两部分:加入视频暂定功能.界面规范化. 一 视频暂停功能 严格来说这个视频暂定功能算是视频人脸性别识别的一个遗留问题,本 ...
- C++开发人脸性别识别教程(12)——加入性别识别功能
经过之前几篇博客的解说,我们已经成功搭建了MFC应用框架,并实现了主要的图像显示和人脸检測程序,在这篇博文中我们要向当中加入性别识别代码. 关于性别识别,之前已经专门拿出两篇博客的篇幅来进行解说.这里 ...
- C++开发人脸性别识别教程(3)——OpenCv配置和ImageWatch插件介绍
OpenCv是C++图像处理的重要工具.这个人脸性别识别的项目就是借助OpenCv进行开发的. 尽管网上已经有了非常多关于OpenCv的配置教程,但出于教程完整性考虑.这里还是用专门的一篇博客来介绍O ...
- C++开发人脸性别识别教程(5)——通过FaceRecognizer类实现性别识别
在之前的博客中已经攻克了人脸检測的问题,我们计划在这篇博客中介绍人脸识别.性别识别方面的相关实现方法. 事实上性别识别和人脸识别本质上是相似的,由于这里仅仅是一个简单的MFC开发,主要工作并不在算法研 ...
- C++开发人脸性别识别教程(7)——搭建MFC框架之界面绘制
在之前的博客中我们已经将项目中用到的算法表述完成,包含人脸检測算法以及四种性别识别算法,在这篇博客中我们将着手搭建主要的MFC框架. 一.框架概况 在这篇博文中我们将搭建最主要的MFC框架.绘制MFC ...
- C++开发人脸性别识别教程(9)——搭建MFC框架之显示图片
在之前的博客中我们已经实现读取用户选定的目录.并将其路径保存在对应的变量中.在这篇博文中我们将介绍怎样借助CvvImage类将图片显示在picture控件中,并自己主动读取目录下的其它图片. 一.加入 ...
- C++开发人脸性别识别教程(6)——通过SVM实现性别识别
http://blog.csdn.net/u013088062/article/details/50480518
- C++开发人脸性别识别教程(8)——搭建MFC框架之读取目录信息
在上一篇博客中我们已经绘制了MFC界面,在这篇博客中我们将加入响应代码,为MFC框架加入一个最主要的功能:打开一个目录. 一.加入相关头文件 这里头文件主要包括三类:opencv头文件.批量读取文件相 ...
随机推荐
- 2019.05.08 《Linux驱动开发入门与实战》
第六章:字符设备 申请设备号---注册设备 1.字符设备的框架: 2.结构体,struct cdev: 3.字符设备的组成: 4.例子: 5.申请和释放设备号: 设备号和设备节点是什么关系.? 设备驱 ...
- LuoguP4012 深海机器人问题(费用流)
题目描述 深海资源考察探险队的潜艇将到达深海的海底进行科学考察. 潜艇内有多个深海机器人.潜艇到达深海海底后,深海机器人将离开潜艇向预定目标移动. 深海机器人在移动中还必须沿途采集海底生物标本.沿途生 ...
- IntelliJ IDEA如何导入maven结构的web工程
第一步:打开一个现有(也可以不打开,直接用import选择Maven类型)的IntelliJ IDEA工程,点击菜单的"File"->"new"-> ...
- 解决 Visual Studio 2013、2015、2017 工具箱不显示ArcGIS 10.2 控件,及ArcGIS模板丢失问题
1.重装ArcObject SDK for .NET Framework方法 (1)问题描述: 环境:WIN10 64bit.Visual Studio 2013.ArcGIS10.1.ArcGIS ...
- mahout用到的典型测试数据集
http://archive.ics.uci.edu/ml/databases/synthetic_control/ 继续
- 根据点画线java
package com.yang; import java.awt.Color; import java.awt.Graphics; import java.util.ArrayList; impor ...
- python处理文件
打开文件: open是内建函数,一个方法 open("test.txt","r",buffering=1) test.txt 表示被打开的文件名,如果不 ...
- Moodle 中文 API 之 文件管理API
File API 文件管理 文件夹 1. 概述 2. 文件域 2.1 命名文件域 3. 提供文件给用户 4. 从用户那获取文件 5. 样例 5.1 浏览文件 5.2 移动文件 5.3 文件列表 5. ...
- @property 和@synthesize
xcode4.4之后,@property包括了@synthesize的功能. 这是编译器的升级. @property有几个作用:1)默认生成一个私有成员变量,并有一个带下划线的别名如_age 2) ...
- Linux系列-安装经常使用软件
安装JDK: 理论篇: 一.下载JDK 二.安装 ①复制到/usr/java/路径下 [plain] view plaincopy #mkdir /usr/java/ #cp jdk-7u25-lin ...