实现人脸识别 (vs2013+opencv3.0)

脸部识别实现(vs2013+opencv3.0)

第一种实现

  1. 这种是通过研究源码 更改而来,第二种是第一种的简化,代码少了很多,而且逻辑比较清楚;具体的实现在代码里面有注释,虽然是英文的,还是比较简单,需要注意的一个api就是:cascade.detectMultiScale()

  2. cascade.detectMultiScale()理解:

            #include "opencv2/objdetect.hpp"
            #include "opencv2/imgcodecs.hpp"
            #include "opencv2/videoio.hpp"
            #include "opencv2/highgui.hpp"
            #include "opencv2/imgproc.hpp"
            #include "opencv2/core/utility.hpp"
            
            #include "opencv2/videoio/videoio_c.h"
            #include "opencv2/highgui/highgui_c.h"
            
            #include <cctype>
            #include <iostream>
            #include <iterator>
            #include <stdio.h>
            
            using namespace std;
            using namespace cv; 
            void detectAndDraw(Mat& img, CascadeClassifier& cascade,
            CascadeClassifier& nestedCascade,
            double scale, bool tryflip);
        
        //要使用到的两个cascade文件,用于联合检测人脸,opencv自带了多个xml文件,在opencv安装目录下的/source/data/haarcascades/ 文件夹内
        string cascadeName = "D:\\opencv\\opencv\\sources\\data\\haarcascades\\haarcascade_frontalface_alt.xml";
        //D:\opencv\opencv\sources\data\haarcascades
        string nestedCascadeName = "D:\\opencv\\opencv\\sources\\data\\haarcascades\\haarcascade_eye_tree_eyeglasses.xml";//检测眼睛
        
        int main()
        {
            CvCapture* capture = 0;
            Mat frame, frameCopy, image;
            CascadeClassifier cascade, nestedCascade;//D:\\code\\opencv\\face\\face\\face\\2.jpg
            char *srcImageFile = "D:\\code\\opencv\\face\\face\\face\\2.jpg";//测试图片
            double scale = 1.3;
            if (!cascade.load(cascadeName))//载入cascade文件
            {
                cerr << "ERROR: Could not load classifier cascade" << endl;
                return -1;
            }
        
            cv::VideoCapture camera(CV_CAP_ANY);
            if (!camera.isOpened())
                return -1;
        
            // 得到帧率  
            double rate = camera.get(CV_CAP_PROP_FPS);
        
            cvNamedWindow("result", 1);
            while (true)
            {
                if (!camera.read(image))
                    break;
                detectAndDraw(image, cascade, nestedCascade, scale, 0);//检测人脸
                int c = waitKey(20);
                if ((char)c == 27)break;
            }
            //以下部分为实现加载图片然后检测
            //image = imread(srcImageFile);
            //cout << "In image read" << endl;
            //if (!image.empty())
            //{
            //  detectAndDraw(image, cascade, nestedCascade, scale, 0);//检测人脸
            //  waitKey(0);
            //}
            cvDestroyWindow("result");
            camera.release();
            return 0;
        }
        
        void detectAndDraw(Mat& img, CascadeClassifier& cascade,CascadeClassifier& nestedCascade,double scale, bool tryflip)
        {
            int i = 0;
            double t = 0;
            vector<Rect> faces, faces2;
            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, smallImg(cvRound(img.rows / scale), cvRound(img.cols / scale), CV_8UC1);
        
            cvtColor(img, gray, COLOR_BGR2GRAY);
            resize(gray, smallImg, smallImg.size(), 0, 0, INTER_LINEAR);
            equalizeHist(smallImg, smallImg);
        
            t = (double)cvGetTickCount();
            cascade.detectMultiScale(smallImg, faces,
                1.1, 2, 0
                //|CASCADE_FIND_BIGGEST_OBJECT
                //|CASCADE_DO_ROUGH_SEARCH
                | CASCADE_SCALE_IMAGE,
                Size(30, 30));
            if (tryflip)
            {
                flip(smallImg, smallImg, 1);
                cascade.detectMultiScale(smallImg, faces2,
                    1.1, 2, 0
                    //|CASCADE_FIND_BIGGEST_OBJECT
                    //|CASCADE_DO_ROUGH_SEARCH
        
                    | CASCADE_SCALE_IMAGE,
                    Size(30, 30));
                for (vector<Rect>::const_iterator r = faces2.begin(); r != faces2.end(); r++)
                {
                    faces.push_back(Rect(smallImg.cols - r->x - r->width, r->y, r->width, r->height));
                }
            }
            t = (double)cvGetTickCount() - t;
            printf("detection time = %g ms\n", t / ((double)cvGetTickFrequency()*1000.));
            for (vector<Rect>::const_iterator r = faces.begin(); r != faces.end(); r++, i++)
            {
                Mat smallImgROI;
                vector<Rect> nestedObjects;
                Point center;
                Scalar color = colors[i % 8];
                int radius;
        
                double aspect_ratio = (double)r->width / r->height;
                if (0.75 < aspect_ratio && aspect_ratio < 1.3)
                {
                    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, 8, 0);
                }
                else
                    rectangle(img, cvPoint(cvRound(r->x*scale), cvRound(r->y*scale)),
                    cvPoint(cvRound((r->x + r->width - 1)*scale), cvRound((r->y + r->height - 1)*scale)),
                    color, 3, 8, 0);
                if (nestedCascade.empty())
                    continue;
                smallImgROI = smallImg(*r);
                nestedCascade.detectMultiScale(smallImgROI, nestedObjects,
                    1.1, 2, 0
                    //|CASCADE_FIND_BIGGEST_OBJECT
                    //|CASCADE_DO_ROUGH_SEARCH
                    //|CASCADE_DO_CANNY_PRUNIN
        
                    | CASCADE_SCALE_IMAGE
                    ,
                    Size(30, 30));
                for (vector<Rect>::const_iterator nr = nestedObjects.begin(); nr != nestedObjects.end(); nr++)
                {
                    center.x = cvRound((r->x + nr->x + nr->width*0.5)*scale);
                    center.y = cvRound((r->y + nr->y + nr->height*0.5)*scale);
                    radius = cvRound((nr->width + nr->height)*0.25*scale);
                    circle(img, center, radius, color, 3, 8, 0);
                }
            }
            cv::imshow("result", img);
        }
    

第二种实现形式

    #include "opencv2/objdetect.hpp"
    #include "opencv2/highgui.hpp"
    #include "opencv2/imgproc.hpp"
    
    #include <iostream>
    #include <stdio.h>
    
    using namespace std;
    using namespace cv;
    
    
    void detectAndDisplay(Mat frame);
    
    String face_cascade_name = "D://opencv//opencv//sources//data//haarcascades//haarcascade_frontalface_alt.xml";
    String eyes_cascade_name = "D://opencv//opencv//sources//data//haarcascades//haarcascade_eye_tree_eyeglasses.xml";
    CascadeClassifier face_cascade;
    CascadeClassifier eyes_cascade;
    String window_name = "Capture - Face detection";
    
    int main(void)
    {
        VideoCapture capture;
        Mat frame;
    
    
        if (!face_cascade.load(face_cascade_name)){ printf("--(!) Error loading face casecade\n"); return -1; }
        if (!eyes_cascade.load(eyes_cascade_name)){ printf("--(!) Error loading eyes cascade\n");  return -1;}
    
    
        //cv::VideoCapture camera(CV_CAP_ANY);
        //if (!camera.isOpened())
        //  return -1;
    
        //open the capture ;and pay attation the CV_CAP_ANY,when you input 0.it' failed ,it appears the device parm is not 0,it convient input CV_CAP_ANY;
        capture.open(CV_CAP_ANY);
        if (!capture.isOpened()){printf("-- Error opening video capture \n");return -1;}
        
        //capture read the frame;
        while (capture.read(frame))
        {
            if (frame.empty()){printf(" -- (!)Error No Capture frame --break"); break;}
            detectAndDisplay(frame);
            int c = waitKey(20);
            if ((char)c == 27){ break; }
    
        }
        return 0;
    }
    
    
    void detectAndDisplay(Mat frame)
    {
        std::vector<Rect> faces;
        Mat frame_gray;
    
        //convert to gray image,and equailzehist 
        cvtColor(frame, frame_gray, COLOR_BGR2GRAY);
        equalizeHist(frame_gray, frame_gray);
    
        //detect face
        face_cascade.detectMultiScale(frame_gray, faces, 1.1, 2, 0 | CASCADE_SCALE_IMAGE, Size(30, 30));
    
    
        for (size_t i = 0; i < faces.size(); i++)
        {
            //gain the face center coordinate 
            Point center(faces[i].x + faces[i].width / 2, faces[i].y + faces[i].height / 2);
            ellipse(frame, center, Size(faces[i].width / 2, faces[i].height / 2), 0, 0, 360, Scalar(255, 0, 255), 2, 8, 0);
    
    
            Mat faceROI = frame_gray(faces[i]);
            std::vector<Rect> eyes;
            eyes_cascade.detectMultiScale(faceROI, eyes, 1.1, 2, 0 | CASCADE_SCALE_IMAGE, Size(30, 30));
    
            for (size_t j = 0; j < eyes.size(); j++)
            {
                Point eye_center(faces[i].x + eyes[j].x + eyes[j].width / 2, faces[i].y + eyes[j].y + eyes[j].height / 2);
                int radius = cvRound((eyes[j].width + eyes[j].height)*0.25);
                circle(frame, eye_center, radius, Scalar(255, 0, 0), 4, 8, 0);
            }
        }
        imshow(window_name, frame);
    }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,565评论 6 479
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,021评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,003评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,015评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,020评论 5 370
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,856评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,178评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,824评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,264评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,788评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,913评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,535评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,130评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,102评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,334评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,298评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,622评论 2 343

推荐阅读更多精彩内容