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

AndroidAPI如何实现人脸检测

这篇文章主要介绍AndroidAPI如何实现人脸检测,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!图片来源:Wikipedia所谓

这篇文章主要介绍Android API如何实现人脸检测,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!

图片来源: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。代码如下:

public class TutorialOnFaceDetect1 extends Activity {    private MyImageView mIV;    private Bitmap mFaceBitmap;    private int mFaceWidth = 200;    private int mFaceHeight = 200;    private static final int MAX_FACES = 1;    private static String TAG = "TutorialOnFaceDetect";      @Override   public void onCreate(Bundle savedInstanceState) {   super.onCreate(savedInstanceState);      mIV = new MyImageView(this);   setContentView(mIV, new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT));      // load the photo   Bitmap b = BitmapFactory.decodeResource(getResources(), R.drawable.face3);   mFaceBitmap = b.copy(Bitmap.Config.RGB_565, true);   b.recycle();      mFaceWidth = mFaceBitmap.getWidth();   mFaceHeight = mFaceBitmap.getHeight();   mIV.setImageBitmap(mFaceBitmap);      // perform face detection and set the feature points setFace();      mIV.invalidate();   }      public void setFace() {   FaceDetector fd;   FaceDetector.Face [] faces = new FaceDetector.Face[MAX_FACES];   PointF midpoint = new PointF();   int [] fpx = null;   int [] fpy = null;   int count = 0;      try {   fd = new FaceDetector(mFaceWidth, mFaceHeight, MAX_FACES);   count = fd.findFaces(mFaceBitmap, faces);   } catch (Exception e) {   Log.e(TAG, "setFace(): " + e.toString());   return;   }      // check if we detect any faces   if (count > 0) {   fpx = new int[count];   fpy = new int[count];      for (int i = 0; i < count; i++) {   try {   faces[i].getMidPoint(midpoint);      fpx[i] = (int)midpoint.x;   fpy[i] = (int)midpoint.y;   } catch (Exception e) {   Log.e(TAG, "setFace(): face " + i + ": " + e.toString());   }   }   }      mIV.setDisplayPoints(fpx, fpy, count, 0);   }   }

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

// set up detected face features for display   public void setDisplayPoints(int [] xx, int [] yy, int total, int style) {    mDisplayStyle = style;    mPX = null;    mPY = null;      if (xx != null && yy != null && total > 0) {   mPX = new int[total];   mPY = new int[total];      for (int i = 0; i < total; i++) {   mPX[i] = xx[i];   mPY[i] = yy[i];   }   }   }
Android API如何实现人脸检测

图1:单一人脸检测

多人脸检测

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

private static final int MAX_FACES = 10;

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

Android API如何实现人脸检测

图2:多人人脸检测

定位眼睛中心位置

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

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

public class TutorialOnFaceDetect extends Activity {    private MyImageView mIV;    private Bitmap mFaceBitmap;    private int mFaceWidth = 200;    private int mFaceHeight = 200;    private static final int MAX_FACES = 10;    private static String TAG = "TutorialOnFaceDetect";   private static boolean DEBUG = false;      protected static final int GUIUPDATE_SETFACE = 999;   protected Handler mHandler = new Handler(){   // @Override   public void handleMessage(Message msg) {   mIV.invalidate();      super.handleMessage(msg);   }   };      @Override   public void onCreate(Bundle savedInstanceState) {   super.onCreate(savedInstanceState);      mIV = new MyImageView(this);   setContentView(mIV, new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT));      // load the photo   Bitmap b = BitmapFactory.decodeResource(getResources(), R.drawable.face3);   mFaceBitmap = b.copy(Bitmap.Config.RGB_565, true);   b.recycle();      mFaceWidth = mFaceBitmap.getWidth();   mFaceHeight = mFaceBitmap.getHeight();   mIV.setImageBitmap(mFaceBitmap);   mIV.invalidate();      // perform face detection in setFace() in a background thread   doLengthyCalc();   }      public void setFace() {   FaceDetector fd;   FaceDetector.Face [] faces = new FaceDetector.Face[MAX_FACES];   PointF eyescenter = new PointF();   float eyesdist = 0.0f;   int [] fpx = null;   int [] fpy = null;   int count = 0;      try {   fd = new FaceDetector(mFaceWidth, mFaceHeight, MAX_FACES);   count = fd.findFaces(mFaceBitmap, faces);   } catch (Exception e) {   Log.e(TAG, "setFace(): " + e.toString());   return;   }      // check if we detect any faces   if (count > 0) {   fpx = new int[count * 2];   fpy = new int[count * 2];      for (int i = 0; i < count; i++) {   try {   faces[i].getMidPoint(eyescenter);   eyesdist = faces[i].eyesDistance();      // set up left eye location   fpx[2 * i] = (int)(eyescenter.x - eyesdist / 2);   fpy[2 * i] = (int)eyescenter.y;      // set up right eye location   fpx[2 * i + 1] = (int)(eyescenter.x + eyesdist / 2);   fpy[2 * i + 1] = (int)eyescenter.y;      if (DEBUG) {   Log.e(TAG, "setFace(): face " + i + ": confidence = " + faces[i].confidence()   + ", eyes distance = " + faces[i].eyesDistance()   + ", pose = ("+ faces[i].pose(FaceDetector.Face.EULER_X) + ","   + faces[i].pose(FaceDetector.Face.EULER_Y) + ","   + faces[i].pose(FaceDetector.Face.EULER_Z) + ")"   + ", eyes midpoint = (" + eyescenter.x + "," + eyescenter.y +")");   }   } catch (Exception e) {   Log.e(TAG, "setFace(): face " + i + ": " + e.toString());   }   }   }      mIV.setDisplayPoints(fpx, fpy, count * 2, 1);   }      private void doLengthyCalc() {   Thread t = new Thread() {   Message m = new Message();      public void run() {   try {   setFace();   m.what = TutorialOnFaceDetect.GUIUPDATE_SETFACE;   TutorialOnFaceDetect.this.mHandler.sendMessage(m);   } catch (Exception e) {   Log.e(TAG, "doLengthyCalc(): " + e.toString());   }   }   };      t.start();   }   }
Android API如何实现人脸检测

以上是“Android API如何实现人脸检测”这篇文章的所有内容,感谢各位的阅读!希望分享的内容对大家有帮助,更多相关知识,欢迎关注编程笔记行业资讯频道!


推荐阅读
  • Nginx使用AWStats日志分析的步骤及注意事项
    本文介绍了在Centos7操作系统上使用Nginx和AWStats进行日志分析的步骤和注意事项。通过AWStats可以统计网站的访问量、IP地址、操作系统、浏览器等信息,并提供精确到每月、每日、每小时的数据。在部署AWStats之前需要确认服务器上已经安装了Perl环境,并进行DNS解析。 ... [详细]
  • 本文介绍了闭包的定义和运转机制,重点解释了闭包如何能够接触外部函数的作用域中的变量。通过词法作用域的查找规则,闭包可以访问外部函数的作用域。同时还提到了闭包的作用和影响。 ... [详细]
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • 本文介绍了Oracle数据库中tnsnames.ora文件的作用和配置方法。tnsnames.ora文件在数据库启动过程中会被读取,用于解析LOCAL_LISTENER,并且与侦听无关。文章还提供了配置LOCAL_LISTENER和1522端口的示例,并展示了listener.ora文件的内容。 ... [详细]
  • CF:3D City Model(小思维)问题解析和代码实现
    本文通过解析CF:3D City Model问题,介绍了问题的背景和要求,并给出了相应的代码实现。该问题涉及到在一个矩形的网格上建造城市的情景,每个网格单元可以作为建筑的基础,建筑由多个立方体叠加而成。文章详细讲解了问题的解决思路,并给出了相应的代码实现供读者参考。 ... [详细]
  • Html5-Canvas实现简易的抽奖转盘效果
    本文介绍了如何使用Html5和Canvas标签来实现简易的抽奖转盘效果,同时使用了jQueryRotate.js旋转插件。文章中给出了主要的html和css代码,并展示了实现的基本效果。 ... [详细]
  • mysql-cluster集群sql节点高可用keepalived的故障处理过程
    本文描述了mysql-cluster集群sql节点高可用keepalived的故障处理过程,包括故障发生时间、故障描述、故障分析等内容。根据keepalived的日志分析,发现bogus VRRP packet received on eth0 !!!等错误信息,进而导致vip地址失效,使得mysql-cluster的api无法访问。针对这个问题,本文提供了相应的解决方案。 ... [详细]
  • Oracle seg,V$TEMPSEG_USAGE与Oracle排序的关系及使用方法
    本文介绍了Oracle seg,V$TEMPSEG_USAGE与Oracle排序之间的关系,V$TEMPSEG_USAGE是V_$SORT_USAGE的同义词,通过查询dba_objects和dba_synonyms视图可以了解到它们的详细信息。同时,还探讨了V$TEMPSEG_USAGE的使用方法。 ... [详细]
  • 深入理解Kafka服务端请求队列中请求的处理
    本文深入分析了Kafka服务端请求队列中请求的处理过程,详细介绍了请求的封装和放入请求队列的过程,以及处理请求的线程池的创建和容量设置。通过场景分析、图示说明和源码分析,帮助读者更好地理解Kafka服务端的工作原理。 ... [详细]
  • Activiti7流程定义开发笔记
    本文介绍了Activiti7流程定义的开发笔记,包括流程定义的概念、使用activiti-explorer和activiti-eclipse-designer进行建模的方式,以及生成流程图的方法。还介绍了流程定义部署的概念和步骤,包括将bpmn和png文件添加部署到activiti数据库中的方法,以及使用ZIP包进行部署的方式。同时还提到了activiti.cfg.xml文件的作用。 ... [详细]
  • 本文介绍了在开发Android新闻App时,搭建本地服务器的步骤。通过使用XAMPP软件,可以一键式搭建起开发环境,包括Apache、MySQL、PHP、PERL。在本地服务器上新建数据库和表,并设置相应的属性。最后,给出了创建new表的SQL语句。这个教程适合初学者参考。 ... [详细]
  • IhaveconfiguredanactionforaremotenotificationwhenitarrivestomyiOsapp.Iwanttwodiff ... [详细]
  • Linux重启网络命令实例及关机和重启示例教程
    本文介绍了Linux系统中重启网络命令的实例,以及使用不同方式关机和重启系统的示例教程。包括使用图形界面和控制台访问系统的方法,以及使用shutdown命令进行系统关机和重启的句法和用法。 ... [详细]
  • 本文介绍了Java工具类库Hutool,该工具包封装了对文件、流、加密解密、转码、正则、线程、XML等JDK方法的封装,并提供了各种Util工具类。同时,还介绍了Hutool的组件,包括动态代理、布隆过滤、缓存、定时任务等功能。该工具包可以简化Java代码,提高开发效率。 ... [详细]
  • [译]技术公司十年经验的职场生涯回顾
    本文是一位在技术公司工作十年的职场人士对自己职业生涯的总结回顾。她的职业规划与众不同,令人深思又有趣。其中涉及到的内容有机器学习、创新创业以及引用了女性主义者在TED演讲中的部分讲义。文章表达了对职业生涯的愿望和希望,认为人类有能力不断改善自己。 ... [详细]
author-avatar
Emily___Emily_622
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有