Demo功能:利用android自带的人脸识别进行识别,标记出眼睛和人脸位置。点击按键后进行人脸识别,完毕后显示到imageview上。

第一部分:布局文件activity_main.xml

 

  1.  
    <RelativeLayout xmlns:android=“http://schemas.android.com/apk/res/android”
  2.  
    xmlns:tools=“http://schemas.android.com/tools”
  3.  
    android:id=“@+id/layout_main”
  4.  
    android:layout_width=“match_parent”
  5.  
    android:layout_height=“match_parent”
  6.  
    android:paddingBottom=“@dimen/activity_vertical_margin”
  7.  
    android:paddingLeft=“@dimen/activity_horizontal_margin”
  8.  
    android:paddingRight=“@dimen/activity_horizontal_margin”
  9.  
    android:paddingTop=“@dimen/activity_vertical_margin”
  10.  
    tools:context=“.MainActivity” >
  11.  
     
  12.  
    <TextView
  13.  
    android:id=“@+id/textview_hello”
  14.  
    android:layout_width=“wrap_content”
  15.  
    android:layout_height=“wrap_content”
  16.  
    android:text=“@string/hello_world” />
  17.  
     
  18.  
    <ImageView
  19.  
    android:id=“@+id/imgview”
  20.  
    android:layout_width=“wrap_content”
  21.  
    android:layout_height=“wrap_content”
  22.  
    android:layout_below=“@id/textview_hello” />
  23.  
     
  24.  
    <Button
  25.  
    android:id=“@+id/btn_detect_face”
  26.  
    android:layout_width=“wrap_content”
  27.  
    android:layout_height=“wrap_content”
  28.  
    android:layout_below=“@id/imgview”
  29.  
    android:layout_centerHorizontal=“true”
  30.  
    android:text=“检测人脸” />
  31.  
     
  32.  
    </RelativeLayout>

注意:ImageView四周的padding由布局文件里的这四句话决定:

 

 

  1.  
    android:paddingBottom=”@dimen/activity_vertical_margin”
  2.  
    android:paddingLeft=”@dimen/activity_horizontal_margin”
  3.  
    android:paddingRight=”@dimen/activity_horizontal_margin”
  4.  
    android:paddingTop=”@dimen/activity_vertical_margin”

而上面的两个margin定义在dimens.xml文件里:

 

 

  1.  
    <resources>
  2.  
     
  3.  
    <!– Default screen margins, per the Android Design guidelines. –>
  4.  
    <dimen name=“activity_horizontal_margin”>16dp</dimen>
  5.  
    <dimen name=“activity_vertical_margin”>16dp</dimen>
  6.  
     
  7.  
    </resources>

这里采用的都是默认的,可以忽略!

 

第二部分:MainActivity.java

 

  1.  
    package org.yanzi.testfacedetect;
  2.  
     
  3.  
    import org.yanzi.util.ImageUtil;
  4.  
    import org.yanzi.util.MyToast;
  5.  
     
  6.  
    import android.app.Activity;
  7.  
    import android.graphics.Bitmap;
  8.  
    import android.graphics.Bitmap.Config;
  9.  
    import android.graphics.BitmapFactory;
  10.  
    import android.graphics.Canvas;
  11.  
    import android.graphics.Color;
  12.  
    import android.graphics.Paint;
  13.  
    import android.graphics.Point;
  14.  
    import android.graphics.PointF;
  15.  
    import android.graphics.Rect;
  16.  
    import android.media.FaceDetector;
  17.  
    import android.media.FaceDetector.Face;
  18.  
    import android.os.Bundle;
  19.  
    import android.os.Handler;
  20.  
    import android.os.Message;
  21.  
    import android.util.DisplayMetrics;
  22.  
    import android.util.Log;
  23.  
    import android.view.Menu;
  24.  
    import android.view.View;
  25.  
    import android.view.View.OnClickListener;
  26.  
    import android.view.ViewGroup;
  27.  
    import android.view.ViewGroup.LayoutParams;
  28.  
    import android.widget.Button;
  29.  
    import android.widget.ImageView;
  30.  
    import android.widget.ProgressBar;
  31.  
    import android.widget.RelativeLayout;
  32.  
     
  33.  
    public class MainActivity extends Activity {
  34.  
    static final String tag = “yan”;
  35.  
    ImageView imgView = null;
  36.  
    FaceDetector faceDetector = null;
  37.  
    FaceDetector.Face[] face;
  38.  
    Button detectFaceBtn = null;
  39.  
    final int N_MAX = 2;
  40.  
    ProgressBar progressBar = null;
  41.  
     
  42.  
    Bitmap srcImg = null;
  43.  
    Bitmap srcFace = null;
  44.  
    Thread checkFaceThread = new Thread(){
  45.  
     
  46.  
    @Override
  47.  
    public void run() {
  48.  
    // TODO Auto-generated method stub
  49.  
    Bitmap faceBitmap = detectFace();
  50.  
    mainHandler.sendEmptyMessage(2);
  51.  
    Message m = new Message();
  52.  
    m.what = 0;
  53.  
    m.obj = faceBitmap;
  54.  
    mainHandler.sendMessage(m);
  55.  
     
  56.  
    }
  57.  
     
  58.  
    };
  59.  
    Handler mainHandler = new Handler(){
  60.  
     
  61.  
    @Override
  62.  
    public void handleMessage(Message msg) {
  63.  
    // TODO Auto-generated method stub
  64.  
    //super.handleMessage(msg);
  65.  
    switch (msg.what){
  66.  
    case 0:
  67.  
    Bitmap b = (Bitmap) msg.obj;
  68.  
    imgView.setImageBitmap(b);
  69.  
    MyToast.showToast(getApplicationContext(), “检测完毕”);
  70.  
    break;
  71.  
    case 1:
  72.  
    showProcessBar();
  73.  
    break;
  74.  
    case 2:
  75.  
    progressBar.setVisibility(View.GONE);
  76.  
    detectFaceBtn.setClickable(false);
  77.  
    break;
  78.  
    default:
  79.  
    break;
  80.  
    }
  81.  
    }
  82.  
     
  83.  
    };
  84.  
    @Override
  85.  
    protected void onCreate(Bundle savedInstanceState) {
  86.  
    super.onCreate(savedInstanceState);
  87.  
    setContentView(R.layout.activity_main);
  88.  
    initUI();
  89.  
    initFaceDetect();
  90.  
    detectFaceBtn.setOnClickListener(new OnClickListener() {
  91.  
     
  92.  
    @Override
  93.  
    public void onClick(View v) {
  94.  
    // TODO Auto-generated method stub
  95.  
    mainHandler.sendEmptyMessage(1);
  96.  
    checkFaceThread.start();
  97.  
     
  98.  
    }
  99.  
    });
  100.  
     
  101.  
     
  102.  
     
  103.  
    }
  104.  
     
  105.  
    @Override
  106.  
    public boolean onCreateOptionsMenu(Menu menu) {
  107.  
    // Inflate the menu; this adds items to the action bar if it is present.
  108.  
    getMenuInflater().inflate(R.menu.main, menu);
  109.  
    return true;
  110.  
    }
  111.  
    public void initUI(){
  112.  
     
  113.  
    detectFaceBtn = (Button)findViewById(R.id.btn_detect_face);
  114.  
    imgView = (ImageView)findViewById(R.id.imgview);
  115.  
    LayoutParams params = imgView.getLayoutParams();
  116.  
    DisplayMetrics dm = getResources().getDisplayMetrics();
  117.  
    int w_screen = dm.widthPixels;
  118.  
    // int h = dm.heightPixels;
  119.  
     
  120.  
    srcImg = BitmapFactory.decodeResource(getResources(), R.drawable.kunlong);
  121.  
    int h = srcImg.getHeight();
  122.  
    int w = srcImg.getWidth();
  123.  
    float r = (float)h/(float)w;
  124.  
    params.width = w_screen;
  125.  
    params.height = (int)(params.width * r);
  126.  
    imgView.setLayoutParams(params);
  127.  
    imgView.setImageBitmap(srcImg);
  128.  
    }
  129.  
     
  130.  
    public void initFaceDetect(){
  131.  
    this.srcFace = srcImg.copy(Config.RGB_565, true);
  132.  
    int w = srcFace.getWidth();
  133.  
    int h = srcFace.getHeight();
  134.  
    Log.i(tag, “待检测图像: w = ” + w + “h = ” + h);
  135.  
    faceDetector = new FaceDetector(w, h, N_MAX);
  136.  
    face = new FaceDetector.Face[N_MAX];
  137.  
    }
  138.  
    public boolean checkFace(Rect rect){
  139.  
    int w = rect.width();
  140.  
    int h = rect.height();
  141.  
    int s = w*h;
  142.  
    Log.i(tag, “人脸 宽w = ” + w + “高h = ” + h + “人脸面积 s = ” + s);
  143.  
    if(s < 10000){
  144.  
    Log.i(tag, “无效人脸,舍弃.”);
  145.  
    return false;
  146.  
    }
  147.  
    else{
  148.  
    Log.i(tag, “有效人脸,保存.”);
  149.  
    return true;
  150.  
    }
  151.  
    }
  152.  
    public Bitmap detectFace(){
  153.  
    // Drawable d = getResources().getDrawable(R.drawable.face_2);
  154.  
    // Log.i(tag, “Drawable尺寸 w = ” + d.getIntrinsicWidth() + “h = ” + d.getIntrinsicHeight());
  155.  
    // BitmapDrawable bd = (BitmapDrawable)d;
  156.  
    // Bitmap srcFace = bd.getBitmap();
  157.  
     
  158.  
    int nFace = faceDetector.findFaces(srcFace, face);
  159.  
    Log.i(tag, “检测到人脸:n = ” + nFace);
  160.  
    for(int i=0; i<nFace; i++){
  161.  
    Face f = face[i];
  162.  
    PointF midPoint = new PointF();
  163.  
    float dis = f.eyesDistance();
  164.  
    f.getMidPoint(midPoint);
  165.  
    int dd = (int)(dis);
  166.  
    Point eyeLeft = new Point((int)(midPoint.x – dis/2), (int)midPoint.y);
  167.  
    Point eyeRight = new Point((int)(midPoint.x + dis/2), (int)midPoint.y);
  168.  
    Rect faceRect = new Rect((int)(midPoint.x – dd), (int)(midPoint.y – dd), (int)(midPoint.x + dd), (int)(midPoint.y + dd));
  169.  
    Log.i(tag, “左眼坐标 x = ” + eyeLeft.x + “y = ” + eyeLeft.y);
  170.  
    if(checkFace(faceRect)){
  171.  
    Canvas canvas = new Canvas(srcFace);
  172.  
    Paint p = new Paint();
  173.  
    p.setAntiAlias(true);
  174.  
    p.setStrokeWidth(8);
  175.  
    p.setStyle(Paint.Style.STROKE);
  176.  
    p.setColor(Color.GREEN);
  177.  
    canvas.drawCircle(eyeLeft.x, eyeLeft.y, 20, p);
  178.  
    canvas.drawCircle(eyeRight.x, eyeRight.y, 20, p);
  179.  
    canvas.drawRect(faceRect, p);
  180.  
    }
  181.  
     
  182.  
    }
  183.  
    ImageUtil.saveJpeg(srcFace);
  184.  
    Log.i(tag, “保存完毕”);
  185.  
     
  186.  
    //将绘制完成后的faceBitmap返回
  187.  
    return srcFace;
  188.  
     
  189.  
    }
  190.  
    public void showProcessBar(){
  191.  
    RelativeLayout mainLayout = (RelativeLayout)findViewById(R.id.layout_main);
  192.  
    progressBar = new ProgressBar(MainActivity.this, null, android.R.attr.progressBarStyleLargeInverse); //ViewGroup.LayoutParams.WRAP_CONTENT
  193.  
    RelativeLayout.LayoutParams params = new RelativeLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT);
  194.  
    params.addRule(RelativeLayout.ALIGN_PARENT_TOP, RelativeLayout.TRUE);
  195.  
    params.addRule(RelativeLayout.CENTER_HORIZONTAL, RelativeLayout.TRUE);
  196.  
    progressBar.setVisibility(View.VISIBLE);
  197.  
    //progressBar.setLayoutParams(params);
  198.  
    mainLayout.addView(progressBar, params);
  199.  
     
  200.  
    }
  201.  
     
  202.  
     
  203.  
    }

关于上述代码,注意以下几点:

 

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);得到人脸中心的坐标。下面这两句话得到左右人眼的坐标:

 

  1.  
    Point eyeLeft = new Point((int)(midPoint.x – dis/2), (int)midPoint.y);
  2.  
    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。代码如下:

 

  1.  
    public void showProcessBar(){
  2.  
    RelativeLayout mainLayout = (RelativeLayout)findViewById(R.id.layout_main);
  3.  
    progressBar = new ProgressBar(MainActivity.this, null, android.R.attr.progressBarStyleLargeInverse); //ViewGroup.LayoutParams.WRAP_CONTENT
  4.  
    RelativeLayout.LayoutParams params = new RelativeLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT);
  5.  
    params.addRule(RelativeLayout.ALIGN_PARENT_TOP, RelativeLayout.TRUE);
  6.  
    params.addRule(RelativeLayout.CENTER_HORIZONTAL, RelativeLayout.TRUE);
  7.  
    progressBar.setVisibility(View.VISIBLE);
  8.  
    //progressBar.setLayoutParams(params);
  9.  
    mainLayout.addView(progressBar, params);
  10.  
     
  11.  
    }

事实上这个ProgressBar视觉效果不是太好,用ProgressDialog会更好。这里只不过是提供动态添加ProgressBar的方法。

 

6、程序中设置了checkFaceThread线程用来检测人脸,mainHandler用来控制UI的更新。这里重点说下Thread的构造方法,这里是模仿源码中打开Camera的方法。如果一个线程只需执行一次,则通过这种方法是最好的,比较简洁。反之,如果这个Thread在执行后需要再次执行或重新构造,不建议用这种方法,建议使用自定义Thread,程序逻辑会更容易 控制。在线程执行完毕后,设置button无法再点击,否则线程再次start便会挂掉。

 

  1.  
    Thread checkFaceThread = new Thread(){
  2.  
     
  3.  
    @Override
  4.  
    public void run() {
  5.  
    // TODO Auto-generated method stub
  6.  
    Bitmap faceBitmap = detectFace();
  7.  
    mainHandler.sendEmptyMessage(2);
  8.  
    Message m = new Message();
  9.  
    m.what = 0;
  10.  
    m.obj = faceBitmap;
  11.  
    mainHandler.sendMessage(m);
  12.  
     
  13.  
    }
  14.  
     
  15.  
    };

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

 

参考文献:

链接1:

链接2:

posted on
2018-07-06 11:37 
无网不进 
阅读(1237
评论(0
编辑 
收藏 
举报

版权声明:本文为hbtmwangjin原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/hbtmwangjin/articles/9272749.html