OpenCV實(shí)現(xiàn)人臉檢測
前段日子,寫了個(gè)人臉檢測的小程序,可以檢測標(biāo)記圖片、視頻、攝像頭中的人臉。效果還行吧,用的是opencv提供人臉庫。至于具體的人臉檢測原理,找資料去啃吧。
環(huán)境:VS2013+OPENCV2.4.10+Win8.1
一、基于對(duì)話框的MFC
首先,新建一個(gè)基于對(duì)話框的MFC應(yīng)用程序,命名為myFaceDetect(取消“安全開發(fā)周期(SDL)檢查”勾選,我自己習(xí)慣取消這個(gè))。
放置Button,設(shè)置Button的ID和Caption。
圖片按鈕——ID:IDC_FACEDETECT
視頻按鈕——ID:IDC_FACEV
攝像頭按鈕——ID:IDC_FACEC
二、添加消息響應(yīng)函數(shù)
為圖片按鈕、視頻按鈕、攝像頭按鈕,在類向?qū)е刑砑酉㈨憫?yīng)函數(shù)。
在圖片按鈕上右鍵,選擇類向?qū)АT贑MyFaceDetectDlg類(對(duì)話框類)下選中BN_CLICKED消息,點(diǎn)擊添加處理程序。其余兩個(gè)按鈕,按同樣操作,添加消息響應(yīng)函數(shù)。
完成上述操作后,獲得對(duì)應(yīng)三個(gè)按鈕的消息響應(yīng)函數(shù)。
void CMyFaceDetectDlg::OnClickedFacedetect()//圖片按鈕 void CMyFaceDetectDlg::OnClickedFacev()//視頻按鈕 void CMyFaceDetectDlg::OnClickedFacec()//攝像頭按鈕
三、人臉檢測實(shí)現(xiàn)
首先,將OpenCV2.4.10+VS2013環(huán)境的配置完成,這個(gè)網(wǎng)上有許多教程。這是我以前寫的一篇配置教程:Visual Studio 2013+OpenCV2.4.10環(huán)境搭建教程
對(duì)話框類的頭文件:MyFaceDetectDlg.h
// MyFaceDetectDlg.h : 頭文件
//
#pragma once
#include <opencv2/objdetect/objdetect.hpp>
#include <opencv2\highgui\highgui.hpp>
#include <opencv2\ml\ml.hpp>
#include <opencv.hpp>
#include "afxwin.h"
using namespace cv;
// CMyFaceDetectDlg 對(duì)話框
class CMyFaceDetectDlg : public CDialogEx
{
// 構(gòu)造
public:
CMyFaceDetectDlg(CWnd* pParent = NULL); // 標(biāo)準(zhǔn)構(gòu)造函數(shù)
// 對(duì)話框數(shù)據(jù)
enum { IDD = IDD_MYFACEDETECT_DIALOG };
protected:
virtual void DoDataExchange(CDataExchange* pDX); // DDX/DDV 支持
// 實(shí)現(xiàn)
protected:
HICON m_hIcon;
HICON m_catIcon;//程序的小貓圖標(biāo)。如果想用默認(rèn)的圖片,可以將其注釋掉。
// 生成的消息映射函數(shù)
virtual BOOL OnInitDialog();
afx_msg void OnPaint();
afx_msg HCURSOR OnQueryDragIcon();
DECLARE_MESSAGE_MAP()
public:
afx_msg void OnClickedFacedetect();
public:
CascadeClassifier cascade;//級(jí)聯(lián)分類器
Mat image;//圖片
double scale;//縮小比例??s小圖片可以加快檢測速度,當(dāng)然加快檢測速度還有其他的方法。
public:
void detectAndDraw(Mat& img, CascadeClassifier& cascade, double scale);//添加的實(shí)現(xiàn)人臉檢測的函數(shù),核心函數(shù)
CButton m_btn;//為了美化按鈕添加對(duì)象,可以注釋掉。
afx_msg void OnClickedFacev();
afx_msg void OnClickedFacec();
afx_msg void OnBnClickedCancel();
};
對(duì)話框類的實(shí)現(xiàn):MyFaceDetectDlg.cpp
// MyFaceDetectDlg.cpp : 實(shí)現(xiàn)文件
//
#include "stdafx.h"
#include "MyFaceDetect.h"
#include "MyFaceDetectDlg.h"
#include "afxdialogex.h"
#include <string>
#ifdef _DEBUG
#define new DEBUG_NEW
#endif
// CMyFaceDetectDlg 對(duì)話框
CMyFaceDetectDlg::CMyFaceDetectDlg(CWnd* pParent /*=NULL*/)
: CDialogEx(CMyFaceDetectDlg::IDD, pParent)
{
m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
m_catIcon = AfxGetApp()->LoadIcon(IDI_ICON4);//加載自己的圖標(biāo)(小貓~)
scale = 1.3;
}
void CMyFaceDetectDlg::DoDataExchange(CDataExchange* pDX)
{
CDialogEx::DoDataExchange(pDX);
DDX_Control(pDX, IDC_FACEDETECT, m_btn);
}
BEGIN_MESSAGE_MAP(CMyFaceDetectDlg, CDialogEx)
ON_WM_PAINT()
ON_WM_QUERYDRAGICON()
ON_BN_CLICKED(IDC_FACEDETECT, &CMyFaceDetectDlg::OnClickedFacedetect)
ON_BN_CLICKED(IDC_FACEV, &CMyFaceDetectDlg::OnClickedFacev)
ON_BN_CLICKED(IDC_FACEC, &CMyFaceDetectDlg::OnClickedFacec)
ON_BN_CLICKED(IDCANCEL, &CMyFaceDetectDlg::OnBnClickedCancel)
END_MESSAGE_MAP()
// CMyFaceDetectDlg 消息處理程序
BOOL CMyFaceDetectDlg::OnInitDialog()
{
CDialogEx::OnInitDialog();
// 設(shè)置此對(duì)話框的圖標(biāo)。 當(dāng)應(yīng)用程序主窗口不是對(duì)話框時(shí),框架將自動(dòng)
// 執(zhí)行此操作
//若不需要自己設(shè)置圖標(biāo),可以將后面所有m_catIcon改成m_hIcon
SetIcon(m_catIcon, TRUE); // 設(shè)置大圖標(biāo)。
SetIcon(m_catIcon, FALSE); // 設(shè)置小圖標(biāo)
//按鈕加載圖片背景
//HBITMAP hbmp1 = ::LoadBitmap(AfxGetInstanceHandle(), MAKEINTRESOURCE(IDB_BITMAP2));
//m_btn.SetBitmap(hbmp1);
// TODO: 在此添加額外的初始化代碼
return TRUE; // 除非將焦點(diǎn)設(shè)置到控件,否則返回 TRUE
}
// 如果向?qū)υ捒蛱砑幼钚』粹o,則需要下面的代碼
// 來繪制該圖標(biāo)。 對(duì)于使用文檔/視圖模型的 MFC 應(yīng)用程序,
// 這將由框架自動(dòng)完成。
void CMyFaceDetectDlg::OnPaint()
{
if (IsIconic())
{
CPaintDC dc(this); // 用于繪制的設(shè)備上下文
SendMessage(WM_ICONERASEBKGND, reinterpret_cast<WPARAM>(dc.GetSafeHdc()), 0);
// 使圖標(biāo)在工作區(qū)矩形中居中
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;
// 繪制圖標(biāo)
dc.DrawIcon(x, y, m_catIcon);
}
else
{
/*改變對(duì)話框背景****若需要默認(rèn)背景,可以刪除*/
CPaintDC dc(this);
CRect rect;
GetClientRect(&rect);
CDC dcBmp;
dcBmp.CreateCompatibleDC(&dc);
CBitmap bmpBackGround;
bmpBackGround.LoadBitmap(IDB_BITMAP4);
BITMAP m_bitmap;
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();
}
}
//當(dāng)用戶拖動(dòng)最小化窗口時(shí)系統(tǒng)調(diào)用此函數(shù)取得光標(biāo)
//顯示。
HCURSOR CMyFaceDetectDlg::OnQueryDragIcon()
{
return static_cast<HCURSOR>(m_catIcon);
}
void CMyFaceDetectDlg::OnClickedFacedetect()
{
// TODO: 在此添加控件通知處理程序代碼
CString filename;
//打開對(duì)話框
CFileDialog OpenDlg(TRUE, NULL, NULL, OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT | OFN_NOCHANGEDIR,
_T("圖片 (*.jpg)|*.jpg|(*.*) |*.*|"), NULL);
if (OpenDlg.DoModal() != IDOK)
{
return;
}
filename = OpenDlg.GetPathName();//獲得文件路徑
/*CString轉(zhuǎn)換*string*/
USES_CONVERSION;
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);//調(diào)用人臉檢測函數(shù)
imshow("人臉檢測", image);
return;
}
void CMyFaceDetectDlg::detectAndDraw(Mat& img, CascadeClassifier& cascade, double scale)
{
/*程序核心函數(shù),檢測標(biāo)記人臉*/
int i = 0;
vector<Rect>faces;//定義一個(gè)容器,保存檢測結(jié)果
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);//轉(zhuǎn)化灰度圖
resize(gray, smallImage, smallImage.size(), 0, 0, INTER_LINEAR);//圖片尺度調(diào)整
equalizeHist(smallImage, smallImage);//直方圖均衡
cascade.detectMultiScale(smallImage, faces);//核心,檢測人臉
for (vector<Rect>::const_iterator r = faces.begin(); r != faces.end(); r++, i++)
{
//利用迭代器,標(biāo)記出人臉位置。
Point center;
Scalar color = colors[i % 8];
int radius;
/*計(jì)算出原圖像中的圓心和半徑。公式很簡單,自己寫一下,就可以理解了*/
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, 3);
}
}
void CMyFaceDetectDlg::OnClickedFacev()
{
// TODO: 在此添加控件通知處理程序代碼
//檢測視頻幀中的人臉
CString filename;
CFileDialog OpenDlg(TRUE, NULL, NULL, OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT | OFN_NOCHANGEDIR,
_T("視頻(*.avi)|*.avi|(*.*)|*.*|"), NULL);
if (OpenDlg.DoModal() != IDOK)
{
return;
}
/*CString轉(zhuǎn)換*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;
}
void CMyFaceDetectDlg::OnClickedFacec()
{
// TODO: 在此添加控件通知處理程序代碼
//檢測攝像頭中的人臉數(shù)據(jù)
const String cascade_name = "./haarcascade_frontalface_alt2.xml";
if (!cascade.load(cascade_name))
{
MessageBox(_T("ERROR:Could not load cascade!"));
return;
}
VideoCapture capture(0);//打開攝像頭
if (!capture.isOpened())
{
MessageBox(_T("ERROR:Could not load capture!"));
return;
}
//double rate = capture.get(CV_CAP_PROP_FPS);
//bool stop(false);
//int delay = 1000 / rate;
int k=0;
while (1)
{
if (!capture.read(image))
break;
detectAndDraw(image, cascade, scale);
imshow("人臉檢測", image);
k=waitkey(10);
if (k=27)//ESC鍵
break;
}
capture.release();
return;
}
void CMyFaceDetectDlg::OnBnClickedCancel()
{
// TODO: 在此添加控件通知處理程序代碼
CDialogEx::OnCancel();
}
三 運(yùn)行程序
視頻和圖片都有測試,一般只要是正臉、清晰的都能檢測圖片。另外,需要將haarcascade_frontalface_alt2.xml文件復(fù)制到程序目錄下。
將文件在目錄opencv\sources\data\haarcascades下。
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持我們。
欄 目:C語言
下一篇:C語言實(shí)現(xiàn)圖的最短路徑Floyd算法
本文標(biāo)題:OpenCV實(shí)現(xiàn)人臉檢測
本文地址:http://www.jygsgssxh.com/a1/Cyuyan/1002.html
您可能感興趣的文章
- 01-10數(shù)據(jù)結(jié)構(gòu)課程設(shè)計(jì)-用棧實(shí)現(xiàn)表達(dá)式求值的方法詳解
- 01-10使用OpenGL實(shí)現(xiàn)3D立體顯示的程序代碼
- 01-10求斐波那契(Fibonacci)數(shù)列通項(xiàng)的七種實(shí)現(xiàn)方法
- 01-10C語言 解決不用+、-、&#215;、&#247;數(shù)字運(yùn)算符做加法
- 01-10使用C++實(shí)現(xiàn)全排列算法的方法詳解
- 01-10用C++實(shí)現(xiàn)DBSCAN聚類算法
- 01-10深入全排列算法及其實(shí)現(xiàn)方法
- 01-10全排列算法的非遞歸實(shí)現(xiàn)與遞歸實(shí)現(xiàn)的方法(C++)
- 01-10用C語言實(shí)現(xiàn)單鏈表的各種操作(一)
- 01-10用C語言實(shí)現(xiàn)單鏈表的各種操作(二)


閱讀排行
本欄相關(guān)
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達(dá)
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對(duì)數(shù)函數(shù)的表達(dá)式 c語言中對(duì)
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求
隨機(jī)閱讀
- 01-10使用C語言求解撲克牌的順子及n個(gè)骰子
- 01-11ajax實(shí)現(xiàn)頁面的局部加載
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 08-05織夢dedecms什么時(shí)候用欄目交叉功能?
- 01-10C#中split用法實(shí)例總結(jié)
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 01-10delphi制作wav文件的方法
- 04-02jquery與jsp,用jquery
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改


