计算机视觉开发在近些年来越发火热,而关于人脸检测或识别等相应功能也成为了大家津津乐道的话题。在智能手机端领域中,人脸识别被广泛用于人脸解锁这项功能中,从简单的2D人脸图像识别,到之后加入的3D结构光技术,人脸识别的精确率在飞速的提升(不过现在android手机厂商为了更接近于100%的手机屏占比,往往并不会加入3D结构光这种很占空间的硬件支持)。
对于开发者而言,要想实现人脸检测等相关功能有两种方法:
本文将介绍使用FaceDetector类来实现一个简单的人脸检测小Demo:
下面开始编码部分:
总体是个线性布局(垂直方向),其中加入一个ImageView来显示选择的图片;再加入一个子线性布局(水平方向),放置两个按钮,一个按钮用来选择图片,另一个按钮则是对图片进行人脸检测。
注意:布局文件中的按钮添加了onClick属性,赋值一个字符串;在程序运行时,若是用户点击了该按钮,那么便会自动回调代码中与字符串同名的方法,此写法有点类似于Qt开发中的“转到槽”
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:gravity="center_vertical"
android:padding="1dp">
<ImageView
android:id="@+id/imageView"
android:layout_width="320dp"
android:layout_height="520dp"
android:layout_gravity="center"
android:background="#ffffff"
android:src="@mipmap/ic_launcher" />
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:orientation="horizontal"
android:paddingTop="6dp">
<Button
android:id="@+id/buttonA"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:onClick="selectPhoto"
android:text="select photo" />
<Button
android:id="@+id/buttonB"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:onClick="detectFace"
android:text="detect face" />
LinearLayout>
LinearLayout>
Intent意为“意图”,是android程序中各组件进行交互的一种重要方式。startActivity()是一种常用的开启activity组件的方法,而这边所讲的startActivityForResult()和startActivity()不同之处在于:startActivityForResult主要用来从activity A跳转到activity B,然后返回activity A,并且获取从activity B中传回来的参数。
理解了作用之后,我们便能将其运用在按钮的点击事件上,根据上文布局文件中按钮A的回调函数名为selectPhoto,这里我们写一个selectPhoto()方法,在方法中创建一个intent对象并设置属性以调用android系统中的图库:
public void selectPhoto(View view) {
// 调用系统的图库
Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
intent.addCategory(Intent.CATEGORY_OPENABLE);
intent.setType("image/*");
startActivityForResult(intent, 0);
}
需要知道的是,startActivityForResult执行的时候会自动调用onActivityResult()方法,所以这里就重写onActivityResult获取选择到的图片Uri并显示到ImageView中:
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
if (data != null) {
Uri uri = data.getData();//获取图片的路径
imageView.setImageURI(uri);//ImageView显示图片
}
super.onActivityResult(requestCode, resultCode, data);
}
Bitmap,意为“位图”,是android.graphics包下的一个用于描述图像的类。位图可以理解为一个画架,把图放到上面然后可以对图片做一些列的处理。位图文件图像显示效果好,但是非压缩格式,需要占用较大的存储空间。
获取了图片的Uri之后,我们就可以用一个Bitmap对象来加载并存储这张图片,通过MediaStore.Images.Media.getBitmap()方法。
定义一个数据成员Bitmap对象:
private Bitmap faceImg;
在onActivityResult中添加faceImg的加载
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
if (data != null) {
Uri uri = data.getData();//获取图片的路径
imageView.setImageURI(uri);//ImageView显示图片
try {
this.faceImg = MediaStore.Images.Media.getBitmap(this.getContentResolver(), uri);
}catch(Exception e){
System.out.println("can not load picture");
}
}
super.onActivityResult(requestCode, resultCode, data);
}
FaceDetector类提供了一个成员方法叫作findFaces,
它的使用有两个参数:
返回值:
在布局文件中,按钮B的回调函数名为detectFace,则在代码中写一个detectFace()方法,创建一个FaceDetector类对象和一个FaceDetector.Face对象数组,并调用findFaces方法检测图片中的人脸:
public void detectFace(View view){
FaceDetector faceDetector = new FaceDetector(faceImg.getWidth(), faceImg.getHeight(), MAX_FACES_COUNT);
FaceDetector.Face[] faceList = new FaceDetector.Face[MAX_FACES_COUNT];
int numOfFaceDetected = faceDetector.findFaces(faceImg, faceList);
System.out.println("检测到的人脸有"+numOfFaceDetected+"张");
}
Canvas是android.graphics包下提供的一个绘图工具类,该类提供了一系列的drawXXX()方法,我们可以利用它在原图上绘制出人脸的矩形区域来标识检测出的人脸。
在Canvas对象的构造函数中,需要传入一个Bitmap对象,接下来一系列Canvas的drawXXX行为都会在这个Bitmap对象上进行,即这个Bitmap对象可以理解为一张“画纸”。
在detectFace()中继续添加代码;
首先,我们定义一张“纯白画纸”的bitmap,上面没有任何内容,然后将它传参给Canvas的构造函数:
Bitmap bitmap = Bitmap.createBitmap(faceImg.getWidth(),faceImg.getHeight(),Bitmap.Config.ARGB_8888);
Canvas canvas = new Canvas(bitmap);
现在我们有了“画纸”,还需要什么呢?对,自然是需要一根“画笔”。而android.graphics包下的Paint类就充当了这个“画笔”的角色。
创建一个Paint对象,并设置一些属性:
Paint paint = new Paint();
paint.setColor(Color.YELLOW); //设置画笔颜色:黄
paint.setStyle(Paint.Style.STROKE); //设置风格:实心
paint.setStrokeWidth(5); //设置画笔粗细程度:5
接下来我们就可以开始画图了。
①将前面我们选择的图片绘制在这个空白Bitmap上:
canvas.drawBitmap(faceImg, 0,0, paint);
通过查阅官方API文档可以了解到,FaceDetector查找人脸的原理是找眼睛。在FaceDetector.Face对象中,可通过eyesDistance()成员方法来返回人脸上的眼距;通过getMidPoint(PointF point)成员方法来获取两眼之间的中心点,保存到传入的参数point中。理解了方法之后,我们就可以利用它来绘制人脸区域的矩形框了。
②遍历FaceDetector.Face对象数组,画出矩形框:
for (int i = 0; i < numOfFaceDetected; i++) {
FaceDetector.Face face = faceList[i];
PointF pointF = new PointF();
face.getMidPoint(pointF); //两眼连线的中心点
float eyesDistance = face.eyesDistance(); //眼距
canvas.drawRect(
(int) (pointF.x - eyesDistance),
(int) (pointF.y - eyesDistance / 2),
(int) (pointF.x + eyesDistance),
(int) (pointF.y + eyesDistance * 3 / 2),
paint);
}
最后,我们调用ImageView的setImageBitmap()方法,将得到的这个绘制完成的bitmap“画纸”,显示到ImageView中:
imageView.setImageBitmap(bitmap);
package com.example.faceid;
import android.graphics.Bitmap;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.PointF;
import android.media.FaceDetector;
import android.net.Uri;
import android.os.Bundle;
import android.app.Activity;
import android.content.Intent;
import android.provider.MediaStore;
import android.view.View;
import android.widget.ImageView;
public class MainActivity extends Activity {
private ImageView imageView;
private final int MAX_FACES_COUNT = 5;
private Bitmap faceImg;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
imageView = (ImageView) findViewById(R.id.imageView);
}
public void selectPhoto(View view) {
// 调用系统的图库
Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
intent.addCategory(Intent.CATEGORY_OPENABLE);
intent.setType("image/*");
startActivityForResult(intent, 0);
}
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
if (data != null) {
Uri uri = data.getData();//获取图片的路径
imageView.setImageURI(uri);//ImageView显示图片
try {
this.faceImg = MediaStore.Images.Media.getBitmap(this.getContentResolver(), uri);
}catch(Exception e){
System.out.println("can not load picture");
}
}
super.onActivityResult(requestCode, resultCode, data);
}
public void detectFace(View view){
FaceDetector faceDetector = new FaceDetector(faceImg.getWidth(), faceImg.getHeight(), MAX_FACES_COUNT);
FaceDetector.Face[] faceList = new FaceDetector.Face[MAX_FACES_COUNT];
int numOfFaceDetected = faceDetector.findFaces(faceImg, faceList);
System.out.println("检测到的人脸有"+numOfFaceDetected+"张");
Bitmap bitmap = Bitmap.createBitmap(faceImg.getWidth(),faceImg.getHeight(),Bitmap.Config.ARGB_8888);
Canvas canvas = new Canvas(bitmap);
Paint paint = new Paint();
paint.setColor(Color.YELLOW); //设置画笔颜色:黄
paint.setStyle(Paint.Style.STROKE); //设置风格:实心
paint.setStrokeWidth(5); //设置画笔粗细程度:5
canvas.drawBitmap(faceImg, 0,0, paint);
for (int i = 0; i < numOfFaceDetected; i++) {
FaceDetector.Face face = faceList[i];
PointF pointF = new PointF();
face.getMidPoint(pointF);
float eyesDistance = face.eyesDistance();
canvas.drawRect(
(int) (pointF.x - eyesDistance),
(int) (pointF.y - eyesDistance / 2),
(int) (pointF.x + eyesDistance),
(int) (pointF.y + eyesDistance * 3 / 2),
paint);
}
imageView.setImageBitmap(bitmap);
}
}
点击select photo按钮,选择一张图片
点击detect face按钮,检测人脸
FaceDetector类所提供的人脸识别,检测等方法终归是有限的,且效果也不是非常好。要想对图像进行更多的细致处理,实现更多的功能,还是使用第三方库比如OpenCV来的实在。
最后呢,就祝各位小伙伴编码愉快吧!(~ ̄▽ ̄)~
作者:严道阳
原文链接:https://blog.csdn.net/qq_41817287/article/details/106698872