opencv实现图片与视频中人脸检测功能

 更新时间:2020年4月25日 17:29  点击:1697

本文实例为大家分享了opencv实现人脸检测功能的具体代码,供大家参考,具体内容如下

第一章:反思与总结

上一篇博客我相信自己将人脸检测中的AdaBoost算法解释的非常清晰了,以及如何训练人脸检测的强分类器:人脸检测中AdaBoost算法详解。事后,自我感觉对这个人脸检测还是不够具体,所以自己抽了一下午的时间用opencv实现图片与视频中的人脸检测,下面是我用vs2013加opencv4.9来实现的。做一下声明,我的代码是参考OpenCV实现人脸检测的一个博客写的,非常感谢这位博主,我学到了很多东西,下面是我一下午实践的总结:

第二章:图片中的人脸检测

啥也不说,先上效果图大笑:

下面是福利图了,图中有志玲姐姐(安静):

可惜没匹配上,很伤心~~~~

有人可能会问这么漂亮的背景图是这么高的,下面是代码~

void CmyFaceDetectDlg::OnPaint() 
{ 
 if (IsIconic()) 
 { 
 CPaintDC dc(this); // 用于绘制的设备上下文 
 
 SendMessage(WM_ICONERASEBKGND, reinterpret_cast<WPARAM>(dc.GetSafeHdc()), 0); 
 
 // 使图标在工作区矩形中居中 
 int cxIcon = GetSystemMetrics(SM_CXICON); 
 int cyIcon = GetSystemMetrics(SM_CYICON); 
 CRect rect; 
 GetClientRect(&rect); 
 int x = (rect.Width() - cxIcon + 1) / 2; 
 int y = (rect.Height() - cyIcon + 1) / 2; 
 
 // 绘制图标 
 dc.DrawIcon(x, y, m_hIcon); 
 } 
 else 
 { 
 /*改变对话框背景****若需要默认背景,可以删除*/ 
 CPaintDC dc(this); 
 CRect rect; 
 GetClientRect(&rect); 
 CDC dcBmp; 
 dcBmp.CreateCompatibleDC(&dc); 
 CBitmap bmpBackGround; 
 bmpBackGround.LoadBitmap(<span style="color:#FF6666;">IDB_BEIJING</span>);//IDB_BEIJING是背景的图片ID,在资源视图中插入资源,选择BITMAP 

BITMAP m_bitmap; //上传图片(BMP)格式,将ID设为一致就好了 
bmpBackGround.GetBitmap(&m_bitmap); 
CBitmap *pbmpOld = dcBmp.SelectObject(&bmpBackGround); 
dc.StretchBlt(0, 0, rect.Width(), rect.Height(), &dcBmp, 0, 0, m_bitmap.bmWidth, m_bitmap.bmHeight, SRCCOPY); 
CDialogEx::OnPaint(); 
}}

好了,下面进入正题,如何实现图片中的人脸匹配,见代码,后面有详细解释:

void CmyFaceDetectDlg::OnBnClickedFacedetect() 
{ 
 // TODO: 在此添加控件通知处理程序代码 
 CString filename; 
 //打开对话框 
 CFileDialog OpenDlg(TRUE, NULL, NULL, OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT | OFN_NOCHANGEDIR, 
 _T("图片 (*.jpg)|*.jpg|(*.*) |*.*|"), NULL); 
 if (OpenDlg.DoModal() != IDOK) 
 { 
 return; 
 } 
 filename = OpenDlg.GetPathName();//获得文件路径 
 /*CString转换*string*/ 
 USES_CONVERSION;//USES_CONVERSION是用来转换类型的 
 //USES_CONVERSION它是在堆栈上分配空间的,也就是说你在你在函数未结束就不会被释放掉。所有要注意不要在一个函数中用while循环执行它,不然栈空间就马上会分配完(栈空间一般只有2M,很小) 
 std::string tempName(W2A(filename));//转换过程 
 image = imread(tempName);//读取图片 
 const String cascade_name = "./haarcascade_frontalface_alt2.xml";//加载人脸库 
 if (!cascade.load(cascade_name)) 
 { 
 MessageBox(_T("ERROR:Could not load cascade!")); 
 return; 
 } 
 if (!image.data) 
 { 
 MessageBox(_T("ERROR:Could not load image!")); 
 return; 
 } 
 namedWindow("人脸检测", CV_WINDOW_AUTOSIZE); 
 detectAndDraw(image, cascade, scale);//调用人脸检测函数 
 imshow("人脸检测", image); 
 return; 
} 
 
void CmyFaceDetectDlg::detectAndDraw(Mat& img, CascadeClassifier& cascade, double scale) 
{ 
 /*程序核心函数,检测标记人脸*/ 
 int i = 0; 
 vector<Rect>faces;//定义一个容器,保存检测结果 
 const static Scalar colors[] = { 
 CV_RGB(0, 0, 255), 
 CV_RGB(0, 128, 255), 
 CV_RGB(0, 255, 255), 
 CV_RGB(0, 255, 0), 
 CV_RGB(255, 128, 0), 
 CV_RGB(255, 255, 0), 
 CV_RGB(255, 0, 0), 
 CV_RGB(255, 0, 255) 
 }; 
 Mat gray, smallImage(cvRound(img.rows / scale), cvRound(img.cols / scale), CV_8UC1);//用cvRound取整 
 cvtColor(img, gray, CV_BGR2GRAY);//转化灰度图 
 resize(gray, smallImage, smallImage.size(), 0, 0, INTER_LINEAR);//图片尺度调整,将gray调整为smallImage.size大小,方法为INTER_LINEAR:局部像素的重采样 
 equalizeHist(smallImage, smallImage);//直方图均衡 
 cascade.detectMultiScale(smallImage, faces);//核心,检测人脸 
 //const_iterator迭代器,是不能改变r所指向的元素的值的 
 for (vector<Rect>::const_iterator r = faces.begin(); r != faces.end(); r++, i++) 
 { 
 //利用迭代器,标记出人脸位置。 
 Point center; 
 Scalar color = colors[i % 8]; 
 int radius; 
 /*计算出原图像中的圆心和半径。公式很简单,自己写一下,就可以理解了*/ 
 center.x = cvRound((r->x + r->width*0.5)*scale); 
 center.y = cvRound((r->y + r->height*0.5)*scale); 
 radius = cvRound((r->width + r->height)*0.25*scale); 
 circle(img, center, radius, color, 2); 
 } 
} 

注意我是在一个MFC的对话框中,这个界面图中按下“图片”button后的操作。

第三章:视频中的人脸检测

其实,和图片中的原理是一样的。因为视频又一帧一帧的图片组成,我们设定一个短的时间间隔,就可以更图片一样了。
先看效果吧:(说明,该视频是一个女子在跳芭蕾舞,我截去3张图片来达到以点概面的效果)

下面见代码:

void CmyFaceDetectDlg::OnBnClickedFacev() 
{ 
 // TODO: 在此添加控件通知处理程序代码 
 //检测视频帧中的人脸 
 CString filename; 
 CFileDialog OpenDlg(TRUE, NULL, NULL, OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT | OFN_NOCHANGEDIR, 
 _T("视频(*.avi)|*.avi|(*.*)|*.*|"), NULL); 
 if (OpenDlg.DoModal() != IDOK) 
 { 
 return; 
 } 
 /*CString转换*string*/ 
 filename = OpenDlg.GetPathName(); 
 USES_CONVERSION; 
 std::string tempName(W2A(filename)); 
 const String cascade_name = "./haarcascade_frontalface_alt2.xml"; 
 if (!cascade.load(cascade_name)) 
 { 
 MessageBox(_T("ERROR:Could not load cascade!")); 
 return; 
 } 
 VideoCapture capture(tempName);//打开视频 
 if (!capture.isOpened()) 
 { 
 MessageBox(_T("ERROR:Could not load Video!")); 
 return; 
 } 
 double rate = capture.get(CV_CAP_PROP_FPS); 
 bool stop(false); 
 int delay = 1000 / rate; 
 while (!stop) 
 { 
 if (!capture.read(image))//读取视频帧 
  break; 
 detectAndDraw(image, cascade, scale); 
 imshow("人脸检测", image); 
 if (waitKey(delay) >= 0) 
  stop = true; 
 } 
 capture.release(); 
 return; 
}

第四章:总结

人脸匹配最总要的是如何生成匹配库,也是检测的方法的差别。库的生成和机器学习密切相关,学习永无止境,努力吧!

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持猪先飞。

[!--infotagslink--]

相关文章

  • python opencv 画外接矩形框的完整代码

    这篇文章主要介绍了python-opencv-画外接矩形框的实例代码,代码简单易懂,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下...2021-09-04
  • python opencv通过4坐标剪裁图片

    图片剪裁是常用的方法,那么如何通过4坐标剪裁图片,本文就详细的来介绍一下,感兴趣的小伙伴们可以参考一下...2021-06-04
  • OpenCV如何去除图片中的阴影的实现

    这篇文章主要介绍了OpenCV如何去除图片中的阴影的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-03-29
  • 解决使用OpenCV中的imread()内存报错问题

    这篇文章主要介绍了解决使用OpenCV中的imread()内存报错问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2021-03-16
  • python OpenCV学习笔记

    这篇文章主要介绍了python OpenCV的相关资料,帮助大家更好的理解和学习使用python的opencv,感兴趣的朋友可以了解下...2021-03-31
  • 使用OpenCV去除面积较小的连通域

    这篇文章主要介绍了使用OpenCV去除面积较小的连通域,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-07-05
  • 详解pycharm的python包opencv(cv2)无代码提示问题的解决

    这篇文章主要介绍了详解pycharm的python包opencv(cv2)无代码提示问题的解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-01-29
  • 在C#中使用OpenCV(使用OpenCVSharp)的实现

    这篇文章主要介绍了在C#中使用OpenCV(使用OpenCVSharp)的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-11-15
  • C++ opencv实现车道线识别

    这篇文章主要为大家详细介绍了C++ opencv实现车道线识别,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2021-02-20
  • 详解opencv中画圆circle函数和椭圆ellipse函数

    这篇文章主要介绍了opencv中画圆circle函数和椭圆ellipse函数,本文通过实例代码给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下...2020-05-09
  • Opencv LBPH人脸识别算法详解

    这篇文章主要为大家详细介绍了Opencv LBPH人脸识别算法的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2020-04-25
  • python中的opencv和PIL(pillow)转化操作

    这篇文章主要介绍了python中的opencv和PIL(pillow)转化操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2021-03-16
  • python基于opencv检测程序运行效率

    这篇文章主要介绍了python基于opencv检测程序运行效率,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下...2020-05-09
  • 浅谈OpenCV中的新函数connectedComponentsWithStats用法

    这篇文章主要介绍了浅谈OpenCV中的新函数connectedComponentsWithStats用法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-07-05
  • Opencv实现绿幕视频背景替换功能

    这篇文章主要为大家详细介绍了Opencv实现绿幕视频背景替换功能,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2020-04-25
  • Opencv图像处理之详解掩膜mask

    这篇文章主要介绍了Opencv图像处理之详解掩膜mask,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-03-21
  • OpenCV-Python实现凸包的获取

    凸包是一个计算几何中的概念,在图像处理过程中,我们常常需要寻找图像中包围某个物体的凸包,本文就使用OpenCV实现,感兴趣的可以了解一下...2021-06-08
  • python基于OpenCV模板匹配识别图片中的数字

    这篇文章主要介绍了python基于OpenCV模板匹配识别图片中的数字,帮助大家更好的理解和学习使用python,感兴趣的朋友可以了解下...2021-03-31
  • python-OpenCV 实现将数组转换成灰度图和彩图

    今天小编就为大家分享一篇python-OpenCV 实现将数组转换成灰度图和彩图,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-04-27
  • 使用opencv识别图像红色区域,并输出红色区域中心点坐标

    这篇文章主要介绍了使用opencv识别图像红色区域,并输出红色区域中心点坐标,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-06-03