Android静态图片人脸识别的完整demo(附完整源码)
Demo功能:利用android自带的人脸识别进行识别,标记出眼睛和人脸位置。点击按键后进行人脸识别,完毕后显示到imageview上。
第一部分:布局文件activity_main.xml
- <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
- xmlns:tools="http://schemas.android.com/tools"
- android:id="@+id/layout_main"
- android:layout_width="match_parent"
- android:layout_height="match_parent"
- android:paddingBottom="@dimen/activity_vertical_margin"
- android:paddingLeft="@dimen/activity_horizontal_margin"
- android:paddingRight="@dimen/activity_horizontal_margin"
- android:paddingTop="@dimen/activity_vertical_margin"
- tools:context=".MainActivity" >
- <TextView
- android:id="@+id/textview_hello"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:text="@string/hello_world" />
- <ImageView
- android:id="@+id/imgview"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:layout_below="@id/textview_hello" />
- <Button
- android:id="@+id/btn_detect_face"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:layout_below="@id/imgview"
- android:layout_centerHorizontal="true"
- android:text="检测人脸" />
- </RelativeLayout>
注意:ImageView四周的padding由布局文件里的这四句话决定:
- android:paddingBottom="@dimen/activity_vertical_margin"
- android:paddingLeft="@dimen/activity_horizontal_margin"
- android:paddingRight="@dimen/activity_horizontal_margin"
- android:paddingTop="@dimen/activity_vertical_margin"
而上面的两个margin定义在dimens.xml文件里:
- <resources>
- <!-- Default screen margins, per the Android Design guidelines. -->
- <dimen name="activity_horizontal_margin">16dp</dimen>
- <dimen name="activity_vertical_margin">16dp</dimen>
- </resources>
这里采用的都是默认的,可以忽略!
第二部分:MainActivity.java
- package org.yanzi.testfacedetect;
- import org.yanzi.util.ImageUtil;
- import org.yanzi.util.MyToast;
- import android.app.Activity;
- import android.graphics.Bitmap;
- import android.graphics.Bitmap.Config;
- import android.graphics.BitmapFactory;
- import android.graphics.Canvas;
- import android.graphics.Color;
- import android.graphics.Paint;
- import android.graphics.Point;
- import android.graphics.PointF;
- import android.graphics.Rect;
- import android.media.FaceDetector;
- import android.media.FaceDetector.Face;
- import android.os.Bundle;
- import android.os.Handler;
- import android.os.Message;
- import android.util.DisplayMetrics;
- import android.util.Log;
- import android.view.Menu;
- import android.view.View;
- import android.view.View.OnClickListener;
- import android.view.ViewGroup;
- import android.view.ViewGroup.LayoutParams;
- import android.widget.Button;
- import android.widget.ImageView;
- import android.widget.ProgressBar;
- import android.widget.RelativeLayout;
- public class MainActivity extends Activity {
- static final String tag = "yan";
- ImageView imgView = null;
- FaceDetector faceDetector = null;
- FaceDetector.Face[] face;
- Button detectFaceBtn = null;
- final int N_MAX = 2;
- ProgressBar progressBar = null;
- Bitmap srcImg = null;
- Bitmap srcFace = null;
- Thread checkFaceThread = new Thread(){
- @Override
- public void run() {
- // TODO Auto-generated method stub
- Bitmap faceBitmap = detectFace();
- mainHandler.sendEmptyMessage(2);
- Message m = new Message();
- m.what = 0;
- m.obj = faceBitmap;
- mainHandler.sendMessage(m);
- }
- };
- Handler mainHandler = new Handler(){
- @Override
- public void handleMessage(Message msg) {
- // TODO Auto-generated method stub
- //super.handleMessage(msg);
- switch (msg.what){
- case 0:
- Bitmap b = (Bitmap) msg.obj;
- imgView.setImageBitmap(b);
- MyToast.showToast(getApplicationContext(), "检测完毕");
- break;
- case 1:
- showProcessBar();
- break;
- case 2:
- progressBar.setVisibility(View.GONE);
- detectFaceBtn.setClickable(false);
- break;
- default:
- break;
- }
- }
- };
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_main);
- initUI();
- initFaceDetect();
- detectFaceBtn.setOnClickListener(new OnClickListener() {
- @Override
- public void onClick(View v) {
- // TODO Auto-generated method stub
- mainHandler.sendEmptyMessage(1);
- checkFaceThread.start();
- }
- });
- }
- @Override
- public boolean onCreateOptionsMenu(Menu menu) {
- // Inflate the menu; this adds items to the action bar if it is present.
- getMenuInflater().inflate(R.menu.main, menu);
- return true;
- }
- public void initUI(){
- detectFaceBtn = (Button)findViewById(R.id.btn_detect_face);
- imgView = (ImageView)findViewById(R.id.imgview);
- LayoutParams params = imgView.getLayoutParams();
- DisplayMetrics dm = getResources().getDisplayMetrics();
- int w_screen = dm.widthPixels;
- // int h = dm.heightPixels;
- srcImg = BitmapFactory.decodeResource(getResources(), R.drawable.kunlong);
- int h = srcImg.getHeight();
- int w = srcImg.getWidth();
- float r = (float)h/(float)w;
- params.width = w_screen;
- params.height = (int)(params.width * r);
- imgView.setLayoutParams(params);
- imgView.setImageBitmap(srcImg);
- }
- public void initFaceDetect(){
- this.srcFace = srcImg.copy(Config.RGB_565, true);
- int w = srcFace.getWidth();
- int h = srcFace.getHeight();
- Log.i(tag, "待检测图像: w = " + w + "h = " + h);
- faceDetector = new FaceDetector(w, h, N_MAX);
- face = new FaceDetector.Face[N_MAX];
- }
- public boolean checkFace(Rect rect){
- int w = rect.width();
- int h = rect.height();
- int s = w*h;
- Log.i(tag, "人脸 宽w = " + w + "高h = " + h + "人脸面积 s = " + s);
- if(s < 10000){
- Log.i(tag, "无效人脸,舍弃.");
- return false;
- }
- else{
- Log.i(tag, "有效人脸,保存.");
- return true;
- }
- }
- public Bitmap detectFace(){
- // Drawable d = getResources().getDrawable(R.drawable.face_2);
- // Log.i(tag, "Drawable尺寸 w = " + d.getIntrinsicWidth() + "h = " + d.getIntrinsicHeight());
- // BitmapDrawable bd = (BitmapDrawable)d;
- // Bitmap srcFace = bd.getBitmap();
- int nFace = faceDetector.findFaces(srcFace, face);
- Log.i(tag, "检测到人脸:n = " + nFace);
- for(int i=0; i<nFace; i++){
- Face f = face[i];
- PointF midPoint = new PointF();
- float dis = f.eyesDistance();
- f.getMidPoint(midPoint);
- int dd = (int)(dis);
- Point eyeLeft = new Point((int)(midPoint.x - dis/2), (int)midPoint.y);
- Point eyeRight = new Point((int)(midPoint.x + dis/2), (int)midPoint.y);
- Rect faceRect = new Rect((int)(midPoint.x - dd), (int)(midPoint.y - dd), (int)(midPoint.x + dd), (int)(midPoint.y + dd));
- Log.i(tag, "左眼坐标 x = " + eyeLeft.x + "y = " + eyeLeft.y);
- if(checkFace(faceRect)){
- Canvas canvas = new Canvas(srcFace);
- Paint p = new Paint();
- p.setAntiAlias(true);
- p.setStrokeWidth(8);
- p.setStyle(Paint.Style.STROKE);
- p.setColor(Color.GREEN);
- canvas.drawCircle(eyeLeft.x, eyeLeft.y, 20, p);
- canvas.drawCircle(eyeRight.x, eyeRight.y, 20, p);
- canvas.drawRect(faceRect, p);
- }
- }
- ImageUtil.saveJpeg(srcFace);
- Log.i(tag, "保存完毕");
- //将绘制完成后的faceBitmap返回
- return srcFace;
- }
- public void showProcessBar(){
- RelativeLayout mainLayout = (RelativeLayout)findViewById(R.id.layout_main);
- progressBar = new ProgressBar(MainActivity.this, null, android.R.attr.progressBarStyleLargeInverse); //ViewGroup.LayoutParams.WRAP_CONTENT
- RelativeLayout.LayoutParams params = new RelativeLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT);
- params.addRule(RelativeLayout.ALIGN_PARENT_TOP, RelativeLayout.TRUE);
- params.addRule(RelativeLayout.CENTER_HORIZONTAL, RelativeLayout.TRUE);
- progressBar.setVisibility(View.VISIBLE);
- //progressBar.setLayoutParams(params);
- mainLayout.addView(progressBar, params);
- }
- }
关于上述代码,注意以下几点:
1、
在initUI()函数里初始化UI布局,主要是将ImageView的长宽比设置。根据srcImg的长宽比及屏幕的宽度,设置ImageView的宽
度为屏幕宽度,然后根据比率得到ImageView的高。然后将Bitmap设置到ImageView里。一旦设置了ImageView的长和
宽,Bitmap会自动缩放填充进去,所以对Bitmap就无需再缩放了。
2、
initFaceDetect()函数里初始化人脸识别所需要的变量。首先将Bitmap的ARGB格式转换为RGB_565格式,这是android自
带人脸识别要求的图片格式,必须进行此转化:this.srcFace = srcImg.copy(Config.RGB_565, true);
然后实例化这两个变量:
FaceDetector faceDetector = null;
FaceDetector.Face[] face;
faceDetector = new FaceDetector(w, h, N_MAX);
face = new FaceDetector.Face[N_MAX];
FaceDetector就是用来进行人脸识别的类,face是用来存放识别得到的人脸信息。N_MAX是允许的人脸个数最大值。
3、真正的人脸识别在自定义的方法detectFace()里,核心代码:faceDetector.findFaces(srcFace, face)。在识别后,通过Face f = face[i];得到每个人脸f,通过 float dis = f.eyesDistance();得到两个人眼之间的距离,f.getMidPoint(midPoint);得到人脸中心的坐标。下面这两句话得到左右人眼的坐标:
- Point eyeLeft = new Point((int)(midPoint.x - dis/2), (int)midPoint.y);
- Point eyeRight = new Point((int)(midPoint.x + dis/2), (int)midPoint.y);
下面是得到人脸的矩形:
- Rect faceRect = new Rect((int)(midPoint.x - dd), (int)(midPoint.y - dd), (int)(midPoint.x + dd), (int)(midPoint.y + dd));
注意这里Rect的四个参数其实就是矩形框左上顶点的x 、y坐标和右下顶点的x、y坐标。
4、实际应用中发现,人脸识别会发生误判。所以增加函数checkFace(Rect rect)来判断,当人脸Rect的面积像素点太小时则视为无效人脸。这里阈值设为10000,实际上这个值可以通过整个图片的大小进行粗略估计到。
5、为了让用户看到正在识别的提醒,这里动态添加一个ProgressBar。代码如下:
- public void showProcessBar(){
- RelativeLayout mainLayout = (RelativeLayout)findViewById(R.id.layout_main);
- progressBar = new ProgressBar(MainActivity.this, null, android.R.attr.progressBarStyleLargeInverse); //ViewGroup.LayoutParams.WRAP_CONTENT
- RelativeLayout.LayoutParams params = new RelativeLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT);
- params.addRule(RelativeLayout.ALIGN_PARENT_TOP, RelativeLayout.TRUE);
- params.addRule(RelativeLayout.CENTER_HORIZONTAL, RelativeLayout.TRUE);
- progressBar.setVisibility(View.VISIBLE);
- //progressBar.setLayoutParams(params);
- mainLayout.addView(progressBar, params);
- }
事实上这个ProgressBar视觉效果不是太好,用ProgressDialog会更好。这里只不过是提供动态添加ProgressBar的方法。
6、
程序中设置了checkFaceThread线程用来检测人脸,mainHandler用来控制UI的更新。这里重点说下Thread的构造方法,这里是
模仿源码中打开Camera的方法。如果一个线程只需执行一次,则通过这种方法是最好的,比较简洁。反之,如果这个Thread在执行后需要再次执行或重
新构造,不建议用这种方法,建议使用自定义Thread,程序逻辑会更容易
控制。在线程执行完毕后,设置button无法再点击,否则线程再次start便会挂掉。
- Thread checkFaceThread = new Thread(){
- @Override
- public void run() {
- // TODO Auto-generated method stub
- Bitmap faceBitmap = detectFace();
- mainHandler.sendEmptyMessage(2);
- Message m = new Message();
- m.what = 0;
- m.obj = faceBitmap;
- mainHandler.sendMessage(m);
- }
- };
7、看下识别效果:
原图:
识别后:
最后特别交代下,当人眼距离少于100个像素时会识别不出来。如果静态图片尺寸较少,而手机的densityDpi又比较高的话,当图片放在drawable-hdpi文件夹下时会发生检测不到人脸的情况,同样的测试图片放在drawable-mdpi就可以正常检测。原因是不同的文件夹下,Bitmap加载进来后的尺寸大小不一样。
后续会推出Camera里实时检测并绘制人脸框,进一步研究眨眼检测,眨眼控制拍照的demo,敬请期待。如果您觉得笔者在认真的写博客,请为我投上一票。
CSDN2013博客之星评选:
http://vote.blog.csdn.net/blogstaritem/blogstar2013/yanzi1225627
本文demo下载链接:
http://download.csdn.net/detail/yanzi1225627/6783575
参考文献:
Android静态图片人脸识别的完整demo(附完整源码)的更多相关文章
- 用opencv做的静态图片人脸识别
这次给大家分享一个图像识别方面的小项目,主要功能是识别图像中的人脸并根据人脸在图片库找出同一个与它最相似的图片,也就是辨别不同的人. 环境:VS2013+opencv2.4.13 主要是算法:open ...
- Tika结合Tesseract-OCR 实现光学汉字识别(简体、宋体的识别率百分之百)—附Java源码、测试数据和训练集下载地址
OCR(Optical character recognition) —— 光学字符识别,是图像处理的一个重要分支,中文的识别具有一定挑战性,特别是手写体和草书的识别,是重要和热门的科学研究方向.可 ...
- 人脸检测识别,人脸检测,人脸识别,离线检测,C#源码
百度网盘地址 微云地址 使用虹软人工智能开放平台技术开发完成
- [转]Android事件分发机制完全解析,带你从源码的角度彻底理解(上)
Android事件分发机制 该篇文章出处:http://blog.csdn.net/guolin_blog/article/details/9097463 其实我一直准备写一篇关于Android事件分 ...
- 【转】Android事件分发机制完全解析,带你从源码的角度彻底理解(下)
转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/9153761 记得在前面的文章中,我带大家一起从源码的角度分析了Android中Vi ...
- 2017最新修复福运来完整运营中时时彩源码PC+手机版本功能齐全
QQ:1395239152 2017-3.14最新修复福运来完整运营版时时彩源码PC+手机版本功能齐全 使用php+mysql开发,并带有完整数据库.截图!!! 注意哈 带手机版 以下截图均为测 ...
- Android逆向之旅---反编译利器Apktool和Jadx源码分析以及错误纠正
Android逆向之旅---反编译利器Apktool和Jadx源码分析以及错误纠正 http://blog.csdn.net/jiangwei0910410003/article/details/51 ...
- [转]Asp.Net大型项目实践(11)-基于MVC Action粒度的权限管理【续】【源码在这里】(在线demo,全部源码)
本文转自:http://www.cnblogs.com/legendxian/archive/2010/01/25/1655551.html 接上篇Asp.Net大型项目实践(10)-基于MVC Ac ...
- 人脸识别ArcFace C#DEMO 开发应用全过程
手上有一个项目,需要检验使用本程序的,是否本人!因为在程序使用前,我们都已经做过头像现场采集,所以源头呢是不成问题的,那么人脸检测,人脸比对,怎么办呢?度娘了下,目前流行的几个人脸检测,人脸比对核心, ...
随机推荐
- Android动画-补间(Tween)动画
Android动画的两种方式,其中帧动画上篇文章已经讲了,这次主要讲解的就是补间动画,补间动画就是动画业务场景中常用的旋转,平移,缩放,和渐变效果,帧动画是通过轮播动画实现动画效果,补间动画通过在两个 ...
- JNI 详细使用 基础【步骤】
1.定义本地[native]方法.通常情况下,应单独定义一个类来封装所有native方法.native方法相当于一个[接口]中的方法,只有方法声明,没有方法体. 2.在项目根目录下创建[jni文件夹] ...
- 怎么才能成为一名PHP专家?
本文作者Bruno Skvorc是一名资深的Web开发者.在这篇文章里主要是讲述成为一名专业的PHP专家所要经历的过程,以及在这个过程里要如何学习掌握技巧和对工具的舍取.(以下为编译内容) 当阅读各种 ...
- 一键切换皮肤的解决思想及iframe嵌套时寻找下级iframe的方法
项目中有个一键切换皮肤的功能,感觉还不错,记录下,就是各颜色样式设置起来太复杂了,不知道有没有更简便的方法: 1.切换皮肤结构层 <li title="<s:text name= ...
- HP Onboard Administrator 固件升级
HP Onboard Administrator是HP公司服务器的远程管理平台.更新是一个非常简单的过程,可以完全通过办公自动化web管理界面. 1. 下载所需二进制文件 下载地址:HP BladeS ...
- 【HTML5】实现QQ聊天气泡效果
今天自己用 HTML/CSS 做了个类似QQ的聊天气泡,以下是效果图: 以下说下关键地方的样式设置.然后贴出html和css代码(不多). 步骤1:布局 消息採用div+float布局,每条消息用一个 ...
- js数组对象深度复制
var deepCopy = function(o) { if (o instanceof Array) { var n = []; for (var i = 0; i < o.length; ...
- PHP经典项目案例-(一)博客管理系统5
本篇实现发表博客. 八.发表博客 (1).界面实现file.php <tr> <td colSpan=3 valign="baseline" style ...
- OSX:不同OSX版本号的标记可能不兼容-续
不同OSX版本号的标记可能不兼容-续: 经过測试,10.10DP2的Update.俗称DP3.的版本号也没有纠正这个问题.而造成该问题的是安装过程中一開始就选择中文,假设安装时使用英文.在第一次进入操 ...
- Android 四大组件之 Activity(二)
1.综述 Activity是Android四大组件(Application Components)之一,简单来说Activity就是平常所见到的用户界面,一般情况下,一个Activity所占的窗口是满 ...