人脸检测方法有许多,比如opencv自带的人脸Haar特征分类器和dlib人脸检测方法等。

对于opencv的人脸检测方法,优点是简单,快速;存在的问题是人脸检测效果不好。正面/垂直/光线较好的人脸,该方法可以检测出来,而侧面/歪斜/光线不好的人脸,无法检测。因此,该方法不适合现场应用。而对于dlib人脸检测方法采用64个特征点检测,效果会好于opencv的方法识别率会更高,本文会分别采用这几种方法来实现人脸识别。那个算法更好,跑跑代码就知道。

实时图像捕获

首先在进行人脸识别之前需要先来学点OpenCV的基础,起码知道如何从摄像头获取当前拍到的图像吧。OpenCV其实很简单,接下来的代码就是最基本的起步点。

第一步:打开本机上的摄像头,实例化VideoCapture

  1. camera = cv2.VideoCapture(0)

开始第一帧图像的捕获,这个方法用来测试当前的摄像头是否可用

  1. success, frame = camera.read()

success返回真时表示开始捕捉图像,反则表示摄像头打开失败,接下来就用最少的代码来打开摄像头并将当前的图像直接显示到一个窗口上,具体代码结构如下:

  1. # coding=utf-8
  2. # ~/learn_face/cv_base.py
  3. from __future__ import print_function
  4. import cv2
  5. cameraCapture = cv2.VideoCapture(0)
  6. success, frame = cameraCapture.read()
  7. while success and cv2.waitKey(1) == -1:
  8. success, frame = cameraCapture.read()
  9. #TODO:在此处可放置各种对当前每一帧图像的处理
  10. cv2.imshow("Camera", frame)
  11. cameraCapture.release()
  12. cv2.destroyAllWindows()

将上述代码存为opencv_base.py然后在命令行直接运行查看效果:

  1. python opencv_base.py

效果如下:

HAAR 分类器

基于Haar特征的cascade分类器(classifiers) 是Paul Viola和 Michael Jone在2001年,论文”Rapid Object Detection using a Boosted Cascade of Simple Features”中提出的一种有效的物品检测(object detect)方法。它是一种机器学习方法,通过许多正负样例中训练得到cascade方程,然后将其应用于其他图片。

在OpenCV3的源码的data目录中就可以找到已训练好的HAAR算法模型,至HAAR算法的各种细节与理论有兴趣的直接去Google或者百度吧,一搜一大堆。花时间看一堆理论不如直接上代码,由代码直接理解这些复杂理论的应用更适合开发人员,毕竟我们不是数学家。

使用HAAR模型识别图像中的人脸其实只要三步走,即使你对深度网络一点不懂也没关系,再复杂的理论到最终不过是一个方法调用罢了,了解清楚其中的原理就好。

第一步:初始化分类器并载入已训练好的HARR模型:

  1. face_cascade = cv2.CascadeClassifier(r'haarcascade_frontalface_default.xml')

第二步: 通过cv2.cvtColor方法将当前的图像进行灰度化处理,简化图像的信息:

  1. gray = cv2.cvtColor(frame, cv2.COLOR_BGR2GRAY)

第三步:然后将灰度化后的图像输入到分类器进行预判:

  1. faces = face_cascade.detectMultiScale(gray, 1.3, 5) #识别人脸

只要faces数组的长度大于一就表示检测到当前画面中检测到人脸,反之亦然。简单来说其实人脸检测已经完成,

最后,为了我们可以知道识别出来的结果,我们可以将脸用方框给圈出来,这里写个方法来圈脸:

  1. def mark_face(img,x,y,w,h):
  2. return cv2.rectangle(img, (x, y), (x + w, y + h), (255, 0, 0), 2)

以下为本例的全部代码:

  1. # coding=utf-8
  2. # ~/learn_face/cv_haar.py
  3. from __future__ import print_function
  4. import cv2
  5. def mark_face(img, x, y, w, h):
  6. return cv2.rectangle(img, (x, y), (x + w, y + h), (255, 0, 0), 2)
  7. cameraCapture = cv2.VideoCapture(0)
  8. success, frame = cameraCapture.read()
  9. face_cascade = cv2.CascadeClassifier(r'haarcascade_frontalface_default.xml') # 1.载入模型
  10. while success and cv2.waitKey(1) == -1:
  11. success, frame = cameraCapture.read()
  12. gray = cv2.cvtColor(frame, cv2.COLOR_BGR2GRAY) # 2.生成灰度图
  13. faces = face_cascade.detectMultiScale(
  14. gray,
  15. scaleFactor=1.1,
  16. minNeighbors=5
  17. ) # 3.进行识别
  18. [mark_face(frame, *args) for args in faces] #画出识别的结果
  19. cv2.imshow("Camera", frame)
  20. cameraCapture.release()
  21. cv2.destroyAllWindows()

以下是运行效果:

运行起来就会觉得HAAR的识别效果不怎么样,稍微动一下就很会识别不了。

Dlib

接下来我们试试用DLib这个老牌的专做人脸识别起家的C++库来试试,Dlib是一个跨平台的C++公共库,除了线程支持,网络支持,提供测试以及大量工具等等优点,Dlib还是一个强大的机器学习的C++库,包含了许多机器学习常用的算法。同时支持大量的数值算法如矩阵、大整数、随机数运算等等。Dlib同时还包含了大量的图形模型算法。最重要的是Dlib的文档和例子都非常详细。

与HAAR分类器的检测方法相比dLib就简单得多了,只需要用dlib自带的人脸检测器detector就够了,连模型都省了!之前的代码两步就能完成

第一步:实例化 detector:

  1. detector = dlib.get_frontal_face_detector()

第二步:进行人脸检测

  1. faces = detector(frame, 1)

That's all! 是不是很简单?

以下是本例的全部代码:

  1. # coding=utf-8
  2. # ~/learn_face/cv_dlib.py
  3. from __future__ import print_function
  4. import cv2
  5. import dlib
  6. cameraCapture = cv2.VideoCapture(0)
  7. success, frame = cameraCapture.read()
  8. detector = dlib.get_frontal_face_detector()
  9. while success and cv2.waitKey(1) == -1:
  10. success, frame = cameraCapture.read()
  11. faces = detector(frame, 1)
  12. for k, d in enumerate(faces):
  13. frame = cv2.rectangle(frame, (d.left(), d.top()),
  14. (d.right(), d.bottom()), (255, 0, 0), 2)
  15. cv2.imshow("Camera", frame)
  16. cameraCapture.release()
  17. cv2.destroyAllWindows()

运行上述代码后会发现dlib的效果真的比HAAR的检测效果要好很多!不管头怎么转都能瞬间识别到,画出来的矩形框都不带闪的!

特征点检测

接下来我们用DLib的特征点提取器detector所识别出来的人脸轮廓点给标记出来。关键点(landmarks)提取需要一个特征提取器predictor,为了构建特征提取器,预训练模型必不可少。除了自行进行训练外,可以使用官方提供的一个模型。该模型可从dlib sourceforge 库下载,此模型是从人脸中提出64个特征点进行检测,其准确度相当高。

具体实现思路如下:

  • 第一步:生成灰度图
  • 第二步:生成直方图
  • 第三步:进行检测

以下为全部代码

  1. # coding=utf-8
  2. # ~/learn_face/landmark.py
  3. import cv2
  4. import dlib
  5. cameraCapture = cv2.VideoCapture(0)
  6. success, frame = cameraCapture.read()
  7. detector = dlib.get_frontal_face_detector()
  8. predictor = dlib.shape_predictor(
  9. "shape_predictor_68_face_landmarks.dat")
  10. while success and cv2.waitKey(1) == -1:
  11. success, frame = cameraCapture.read()
  12. gray = cv2.cvtColor(frame, cv2.COLOR_BGR2GRAY) #生成灰度图
  13. clahe = cv2.createCLAHE(clipLimit=2.0, tileGridSize=(8, 8)) #生成直方图
  14. clahe_image = clahe.apply(gray)
  15. detections = detector(clahe_image, 1)
  16. for k, d in enumerate(detections):
  17. shape = predictor(clahe_image, d) # 获取坐标
  18. for i in range(1, 68): # 每张脸都有68个识别点
  19. cv2.circle(frame, (shape.part(i).x, shape.part(i).y), 1, (0, 0, 255),
  20. thickness=2)
  21. cv2.imshow("Camera", frame)
  22. cameraCapture.release()
  23. cv2.destroyAllWindows()

运行效果:

小结

我在macBookPro上跑以上的代码在速度是上没有什么很大区别的,至少不会产生卡顿。但如果换将代码植到树莓3和树莓Zero上区别就明显了,HAAR分类器在树梅Zero上的运行时间平均在1.2s左右,而dlib则需要8s。至于准确率Dlib又明显会优于HAAR。

参考阅读

  • HAAR分类器 - 这篇知乎上的文章对HAAR分类器的原理分析得很详尽,有兴趣可以读一读
  • 本文代码可到我的码云上下载

HAAR与DLib的实时人脸检测之实现与对比的更多相关文章

  1. Python 3 利用 Dlib 实现摄像头实时人脸检测和平铺显示

    1. 引言 在某些场景下,我们不仅需要进行实时人脸检测追踪,还要进行再加工:这里进行摄像头实时人脸检测,并对于实时检测的人脸进行初步提取: 单个/多个人脸检测,并依次在摄像头窗口,实时平铺显示检测到的 ...

  2. Python 3 利用 Dlib 实现摄像头人脸检测特征点标定

    0. 引言 利用 Python 开发,借助 Dlib 库捕获摄像头中的人脸,进行实时人脸 68 个特征点标定: 支持多张人脸: 有截图功能: 图 1 工程效果示例( gif ) 图 2 工程效果示例( ...

  3. 基于Haar特征的Adaboost级联人脸检测分类器

    基于Haar特征的Adaboost级联人脸检测分类器基于Haar特征的Adaboost级联人脸检测分类器,简称haar分类器.通过这个算法的名字,我们可以看到这个算法其实包含了几个关键点:Haar特征 ...

  4. 照片美妆---基于Haar特征的Adaboost级联人脸检测分类器

    原文:照片美妆---基于Haar特征的Adaboost级联人脸检测分类器 本文转载自张雨石http://blog.csdn.net/stdcoutzyx/article/details/3484223 ...

  5. Haar、pico、npd、dlib等多种人脸检测特征及算法结果比较

    原文:opencv.pico.npd.dlib.face++等多种人脸检测算法结果比较 NDP检测结果: 结果分析: Pico(Pixel Intensity Comparison-based Obj ...

  6. 实时人脸检测 (Real-Time Face Detection)

    源地址:http://blog.sina.com.cn/s/blog_79b67dfe0102uzra.html 最近需要用到人脸检测,于是找了篇引用广泛的论文实现了一下:Robust Real-Ti ...

  7. opencv+python实时人脸检测、磨皮

    import numpy as np import cv2 cap = cv2.VideoCapture(0) face_cascade = cv2.CascadeClassifier("d ...

  8. win10+anaconda+cuda配置dlib,使用GPU对dlib的深度学习算法进行加速(以人脸检测为例)

    在计算机视觉和机器学习方向有一个特别好用但是比较低调的库,也就是dlib,与opencv相比其包含了很多最新的算法,尤其是深度学习方面的,因此很有必要学习一下.恰好最近换了一台笔记本,内含一块GTX1 ...

  9. Python 3 利用 Dlib 实现人脸检测和剪切

    0. 引言 利用 Python 开发,借助 Dlib 库进行人脸检测 / face detection 和剪切:   1. crop_faces_show.py : 将检测到的人脸剪切下来,依次排序平 ...

随机推荐

  1. 深圳市共创力推出《以用户为中心的设计UCD方法与实战》课程!

    以用户为中心的设计(UCD)方法与实战 课程特色 现在以市场为中心.科技为基础.体验为卖点的商业社会里,用户体验是赢得用户青睐的关键特性.苹果.google.腾讯等顶级企业的成功充分说明了这一点.如何 ...

  2. pycharm 中按照文档引包方式,引包错误

    * python使用pycharm ide,如果电脑上有多个解释器的,在项目解释器配置的应该是当前使用的解释器: * 可以把当前使用的解释器目录添加到系统环境变量中,这样就不会报错了 另外,如果目录中 ...

  3. TTS 文字转语音 ekho

    1.源码下载 使用svn客户端,执行如下命令下载 svn co https://svn.code.sf.net/p/e-guidedog/code/ 2.官方网站查看说明 http://www.egu ...

  4. Spark应用【根据新df更新旧df】

    // 主键字段保持不变,再转换回来 var columnMap:Map[String, String] = Map() for(key <- keysOpt){ columnMap += (ke ...

  5. mssql sql server 系统更新,如何正确的增加表字段

    转自: http://www.maomao365.com/?p=5277摘要:下文主要讲述,如何对"已上线的系统"中的表,增加新的字段. 系统部署脚本,增加列的方法:在系统脚本发布 ...

  6. java中 this() 和super()的作用及用法

    原文地址:http://www.cnblogs.com/hasse/p/5023392.html 一.This Java关键字this只能用于方法体中.this只能在类中的非静态方法中使用,静态方法和 ...

  7. C语言 大小写字母转换

    //凯鲁嘎吉 - 博客园 http://www.cnblogs.com/kailugaji/ 方法1: #include<stdio.h> #include<stdlib.h> ...

  8. node基础—http模块

    在浏览器输入存在的网址的一个交互过程 1.用户通过浏览器发送一个http的请求到指定的主机 2.服务器接收到该请求,对该请求进行分析和处理 3.服务器处理完成以后,返回对应的数据到用户机器 4.浏览器 ...

  9. Mariadb Redis 的配置使用

    Mariadb Mysql 的配置使用 CentOS 7 Mariadb 的学习 在linux上安装软件的方式 yum安装 在线搜索rpm格式的软件包,进行自动的依赖关系处理,下载,安装 ​ (阿里云 ...

  10. python之模块与包

    一模块 二包 一模块 常见的场景:一个模块就是一个包含了python定义和声明的文件,文件名就是模块名字加上.py的后缀. 但其实import加载的模块分为四个通用类别: 1 使用python编写的代 ...