热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

AndroidAPI教程:人脸检测(FaceDetect)

通过两个主要的API,Android提供了一个直接在位图上进行脸部检测的方法,这两个API分别是   android.media.FaceDetector和android.medi

通过两个主要的API,Android提供了一个直接在位图上进行脸部检测的方法,这两个API分别是    android.media.FaceDetector和android.media.FaceDetector.Face,已经包含在Android官方API中。本教程来自Developer网站,向大家介绍了这些API,同时提供教程中实例代码下载。

Android API教程:人脸检测(Face Detect)

图片来源:Wikipedia

所谓人脸检测就是指从一副图片或者一帧视频中标定出所有人脸的位置和尺寸。人脸检测是人脸识别系统中的一个重要环节,也可以独立应用于视频监控。在数字媒体日益普及的今天,利用人脸检测技术还可以帮助我们从海量图片数据中快速筛选出包含人脸的图片。 在目前的数码相机中,人脸检测可以用来完成自动对焦,即“脸部对焦”。“脸部对焦”是在自动曝光和自动对焦发明后,二十年来最重要的一次摄影技术革新。家用数码相机,占绝大多数的照片是以人为拍摄主体的,这就要求相机的自动曝光和对焦以人物为基准。
via cdstm.cn

构建一个人脸检测的Android Activity

你可以构建一个通用的Android Activity,我们扩展了基类ImageView,成为MyImageView,而我们需要进行检测的包含人脸的位图文件必须是565格式,API才能正常工作。被检测出来的人脸需要一个置信测度(confidence measure),这个措施定义在android.media.FaceDetector.Face.CONFIDENCE_THRESHOLD。

最重要的方法实现在setFace(),它将FaceDetector对象实例化,同时调用findFaces,结果存放在faces里,人脸的中点转移到MyImageView。代码如下:

  1. public class TutorialOnFaceDetect1 extends Activity { 
  2.  private MyImageView mIV; 
  3.  private Bitmap mFaceBitmap; 
  4.  private int mFaceWidth = 200
  5.  private int mFaceHeight = 200
  6.  private static final int MAX_FACES = 1
  7.  private static String TAG = "TutorialOnFaceDetect"
  8.  
  9. @Override 
  10. public void onCreate(Bundle savedInstanceState) { 
  11. super.onCreate(savedInstanceState); 
  12.  
  13. mIV = new MyImageView(this); 
  14. setContentView(mIV, new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT)); 
  15.  
  16. // load the photo 
  17. Bitmap b = BitmapFactory.decodeResource(getResources(), R.drawable.face3); 
  18. mFaceBitmap = b.copy(Bitmap.Config.RGB_565, true); 
  19. b.recycle(); 
  20.  
  21. mFaceWidth = mFaceBitmap.getWidth(); 
  22. mFaceHeight = mFaceBitmap.getHeight(); 
  23. mIV.setImageBitmap(mFaceBitmap); 
  24.  
  25. // perform face detection and set the feature points setFace(); 
  26.  
  27. mIV.invalidate(); 
  28.  
  29. public void setFace() { 
  30. FaceDetector fd; 
  31. FaceDetector.Face [] faces = new FaceDetector.Face[MAX_FACES]; 
  32. PointF midpoint = new PointF(); 
  33. int [] fpx = null
  34. int [] fpy = null
  35. int count = 0
  36.  
  37. try { 
  38. fd = new FaceDetector(mFaceWidth, mFaceHeight, MAX_FACES); 
  39. count = fd.findFaces(mFaceBitmap, faces); 
  40. catch (Exception e) { 
  41. Log.e(TAG, "setFace(): " + e.toString()); 
  42. return
  43.  
  44. // check if we detect any faces 
  45. if (count > 0) { 
  46. fpx = new int[count]; 
  47. fpy = new int[count]; 
  48.  
  49. for (int i = 0; i < count; i++) { 
  50. try { 
  51. faces[i].getMidPoint(midpoint); 
  52.  
  53. fpx[i] = (int)midpoint.x; 
  54. fpy[i] = (int)midpoint.y; 
  55. catch (Exception e) { 
  56. Log.e(TAG, "setFace(): face " + i + ": " + e.toString()); 
  57.  
  58. mIV.setDisplayPoints(fpx, fpy, count, 0); 

接下来的代码中,我们在MyImageView中添加setDisplayPoints() ,用来在被检测出的人脸上标记渲染。图1展示了一个标记在被检测处的人脸上处于中心位置。

  1. // set up detected face features for display 
  2. public void setDisplayPoints(int [] xx, int [] yy, int total, int style) { 
  3.  mDisplayStyle = style; 
  4.  mPX = null
  5.  mPY = null
  6.  
  7. if (xx != null && yy != null && total > 0) { 
  8. mPX = new int[total]; 
  9. mPY = new int[total]; 
  10.  
  11. for (int i = 0; i < total; i++) { 
  12. mPX[i] = xx[i]; 
  13. mPY[i] = yy[i]; 

Android API教程:人脸检测(Face Detect)

图1:单一人脸检测

多人脸检测

通过FaceDetector可以设定检测到人脸数目的上限。比如设置最多只检测10张脸:

  1. private static final int MAX_FACES = 10

图2展示检测到多张人脸的情况。

Android API教程:人脸检测(Face Detect)

图2:多人人脸检测

定位眼睛中心位置

Android人脸检测返回其他有用的信息,例同时会返回如eyesDistance,pose,以及confidence。我们可以通过eyesDistance来定位眼睛的中心位置。

下面的代码中,我们将setFace()放在doLengthyCalc()中。同时图3展示了定位眼睛中心位置的效果。

  1. public class TutorialOnFaceDetect extends Activity { 
  2.  private MyImageView mIV; 
  3.  private Bitmap mFaceBitmap; 
  4.  private int mFaceWidth = 200
  5.  private int mFaceHeight = 200
  6.  private static final int MAX_FACES = 10
  7.  private static String TAG = "TutorialOnFaceDetect"
  8. private static boolean DEBUG = false
  9.  
  10. protected static final int GUIUPDATE_SETFACE = 999
  11. protected Handler mHandler = new Handler(){ 
  12. // @Override 
  13. public void handleMessage(Message msg) { 
  14. mIV.invalidate(); 
  15.  
  16. super.handleMessage(msg); 
  17. }; 
  18.  
  19. @Override 
  20. public void onCreate(Bundle savedInstanceState) { 
  21. super.onCreate(savedInstanceState); 
  22.  
  23. mIV = new MyImageView(this); 
  24. setContentView(mIV, new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT)); 
  25.  
  26. // load the photo 
  27. Bitmap b = BitmapFactory.decodeResource(getResources(), R.drawable.face3); 
  28. mFaceBitmap = b.copy(Bitmap.Config.RGB_565, true); 
  29. b.recycle(); 
  30.  
  31. mFaceWidth = mFaceBitmap.getWidth(); 
  32. mFaceHeight = mFaceBitmap.getHeight(); 
  33. mIV.setImageBitmap(mFaceBitmap); 
  34. mIV.invalidate(); 
  35.  
  36. // perform face detection in setFace() in a background thread 
  37. doLengthyCalc(); 
  38.  
  39. public void setFace() { 
  40. FaceDetector fd; 
  41. FaceDetector.Face [] faces = new FaceDetector.Face[MAX_FACES]; 
  42. PointF eyescenter = new PointF(); 
  43. float eyesdist = 0.0f; 
  44. int [] fpx = null
  45. int [] fpy = null
  46. int count = 0
  47.  
  48. try { 
  49. fd = new FaceDetector(mFaceWidth, mFaceHeight, MAX_FACES); 
  50. count = fd.findFaces(mFaceBitmap, faces); 
  51. catch (Exception e) { 
  52. Log.e(TAG, "setFace(): " + e.toString()); 
  53. return
  54.  
  55. // check if we detect any faces 
  56. if (count > 0) { 
  57. fpx = new int[count * 2]; 
  58. fpy = new int[count * 2]; 
  59.  
  60. for (int i = 0; i < count; i++) { 
  61. try { 
  62. faces[i].getMidPoint(eyescenter); 
  63. eyesdist = faces[i].eyesDistance(); 
  64.  
  65. // set up left eye location 
  66. fpx[2 * i] = (int)(eyescenter.x - eyesdist / 2); 
  67. fpy[2 * i] = (int)eyescenter.y; 
  68.  
  69. // set up right eye location 
  70. fpx[2 * i + 1] = (int)(eyescenter.x + eyesdist / 2); 
  71. fpy[2 * i + 1] = (int)eyescenter.y; 
  72.  
  73. if (DEBUG) { 
  74. Log.e(TAG, "setFace(): face " + i + ": confidence = " + faces[i].confidence() 
  75. ", eyes distance = " + faces[i].eyesDistance() 
  76. ", pose = ("+ faces[i].pose(FaceDetector.Face.EULER_X) + "," 
  77. + faces[i].pose(FaceDetector.Face.EULER_Y) + "," 
  78. + faces[i].pose(FaceDetector.Face.EULER_Z) + ")" 
  79. ", eyes midpoint = (" + eyescenter.x + "," + eyescenter.y +")"); 
  80. catch (Exception e) { 
  81. Log.e(TAG, "setFace(): face " + i + ": " + e.toString()); 
  82.  
  83. mIV.setDisplayPoints(fpx, fpy, count * 21); 
  84.  
  85. private void doLengthyCalc() { 
  86. Thread t = new Thread() { 
  87. Message m = new Message(); 
  88.  
  89. public void run() { 
  90. try { 
  91. setFace(); 
  92. m.what = TutorialOnFaceDetect.GUIUPDATE_SETFACE; 
  93. TutorialOnFaceDetect.this.mHandler.sendMessage(m); 
  94. catch (Exception e) { 
  95. Log.e(TAG, "doLengthyCalc(): " + e.toString()); 
  96. }; 
  97.  
  98. t.start(); 

Android API教程:人脸检测(Face Detect)

图3:定位眼睛中心位置

色彩 vs. 灰度

通常来讲,人脸检测成功取决于搜索人脸高对比度区域,实际效果来看色彩和灰度的差距不会太远。不过很多学者仍在致力于证明色彩比灰度更靠谱。经过在对示例图片的验证,发现Android APIs返回的结果非常接近,似乎APIs意图忽略掉不同颜色通道的因素。请看图4(BTW,独自一人在阴暗环境下请谨慎观看):

Android API教程:人脸检测(Face Detect)

图4:灰度人脸检测看起来会稍微有点恐怖

总结

通过本次教程,我们介绍了简单的Android人脸检测APIs,并通过实例进行了演示。以上的软件包均可在官网上下载,方便大家将其import到Eclipse中。最后提供一些有益的忠告:

  1. 很多应用对人脸检测其实都有着潜在的重要需求,例如去红眼、计算人头数、自动对焦人脸、添加人脸特效等等。
  2. 这个世界上存在有非常多的人脸数据库,有意者请点击此处。
  3. 在实时的人脸检测过程中,Android的表现的会有一点点差强人意。

推荐阅读
  • C# 7.0 新特性:基于Tuple的“多”返回值方法
    本文介绍了C# 7.0中基于Tuple的“多”返回值方法的使用。通过对C# 6.0及更早版本的做法进行回顾,提出了问题:如何使一个方法可返回多个返回值。然后详细介绍了C# 7.0中使用Tuple的写法,并给出了示例代码。最后,总结了该新特性的优点。 ... [详细]
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • 本文介绍了在开发Android新闻App时,搭建本地服务器的步骤。通过使用XAMPP软件,可以一键式搭建起开发环境,包括Apache、MySQL、PHP、PERL。在本地服务器上新建数据库和表,并设置相应的属性。最后,给出了创建new表的SQL语句。这个教程适合初学者参考。 ... [详细]
  • 基于layUI的图片上传前预览功能的2种实现方式
    本文介绍了基于layUI的图片上传前预览功能的两种实现方式:一种是使用blob+FileReader,另一种是使用layUI自带的参数。通过选择文件后点击文件名,在页面中间弹窗内预览图片。其中,layUI自带的参数实现了图片预览功能。该功能依赖于layUI的上传模块,并使用了blob和FileReader来读取本地文件并获取图像的base64编码。点击文件名时会执行See()函数。摘要长度为169字。 ... [详细]
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • 本文介绍了使用Java实现大数乘法的分治算法,包括输入数据的处理、普通大数乘法的结果和Karatsuba大数乘法的结果。通过改变long类型可以适应不同范围的大数乘法计算。 ... [详细]
  • Android中高级面试必知必会,积累总结
    本文介绍了Android中高级面试的必知必会内容,并总结了相关经验。文章指出,如今的Android市场对开发人员的要求更高,需要更专业的人才。同时,文章还给出了针对Android岗位的职责和要求,并提供了简历突出的建议。 ... [详细]
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • 如何实现织梦DedeCms全站伪静态
    本文介绍了如何通过修改织梦DedeCms源代码来实现全站伪静态,以提高管理和SEO效果。全站伪静态可以避免重复URL的问题,同时通过使用mod_rewrite伪静态模块和.htaccess正则表达式,可以更好地适应搜索引擎的需求。文章还提到了一些相关的技术和工具,如Ubuntu、qt编程、tomcat端口、爬虫、php request根目录等。 ... [详细]
  • Monkey《大话移动——Android与iOS应用测试指南》的预购信息发布啦!
    Monkey《大话移动——Android与iOS应用测试指南》的预购信息已经发布,可以在京东和当当网进行预购。感谢几位大牛给出的书评,并呼吁大家的支持。明天京东的链接也将发布。 ... [详细]
  • 本文介绍了Python高级网络编程及TCP/IP协议簇的OSI七层模型。首先简单介绍了七层模型的各层及其封装解封装过程。然后讨论了程序开发中涉及到的网络通信内容,主要包括TCP协议、UDP协议和IPV4协议。最后还介绍了socket编程、聊天socket实现、远程执行命令、上传文件、socketserver及其源码分析等相关内容。 ... [详细]
  • 本文介绍了在Python3中如何使用选择文件对话框的格式打开和保存图片的方法。通过使用tkinter库中的filedialog模块的asksaveasfilename和askopenfilename函数,可以方便地选择要打开或保存的图片文件,并进行相关操作。具体的代码示例和操作步骤也被提供。 ... [详细]
  • Java序列化对象传给PHP的方法及原理解析
    本文介绍了Java序列化对象传给PHP的方法及原理,包括Java对象传递的方式、序列化的方式、PHP中的序列化用法介绍、Java是否能反序列化PHP的数据、Java序列化的原理以及解决Java序列化中的问题。同时还解释了序列化的概念和作用,以及代码执行序列化所需要的权限。最后指出,序列化会将对象实例的所有字段都进行序列化,使得数据能够被表示为实例的序列化数据,但只有能够解释该格式的代码才能够确定数据的内容。 ... [详细]
  • 本文介绍了C#中生成随机数的三种方法,并分析了其中存在的问题。首先介绍了使用Random类生成随机数的默认方法,但在高并发情况下可能会出现重复的情况。接着通过循环生成了一系列随机数,进一步突显了这个问题。文章指出,随机数生成在任何编程语言中都是必备的功能,但Random类生成的随机数并不可靠。最后,提出了需要寻找其他可靠的随机数生成方法的建议。 ... [详细]
  • Metasploit攻击渗透实践
    本文介绍了Metasploit攻击渗透实践的内容和要求,包括主动攻击、针对浏览器和客户端的攻击,以及成功应用辅助模块的实践过程。其中涉及使用Hydra在不知道密码的情况下攻击metsploit2靶机获取密码,以及攻击浏览器中的tomcat服务的具体步骤。同时还讲解了爆破密码的方法和设置攻击目标主机的相关参数。 ... [详细]
author-avatar
不理坏女孩
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有