… …
本文将介绍使用FaceDetector类来实现一个简单的人脸检测小Demo:
-
点击按钮A选择一张本地图片并显示
-
点击按钮B进行对图片中的人脸检测
下面开始编码部分:
总体是个线性布局(垂直方向),其中加入一个ImageView来显示选择的图片;再加入一个子线性布局(水平方向),放置两个按钮,一个按钮用来选择图片,另一个按钮则是对图片进行人脸检测。
注意:布局文件中的按钮添加了onClick属性,赋值一个字符串;在程序运行时,若是用户点击了该按钮,那么便会自动回调代码中与字符串同名的方法,此写法有点类似于Qt开发中的“转到槽”
<?xml version="1.0" encoding="utf-8"?><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” />
布局预览图:
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,
它的使用有两个参数:
-
Bitmap对象,即你需要识别检测的图片
-
一个FaceDetector.Face对象数组,FaceDetector.Face是FaceDetector类中的内部类,提供了对人脸信息的简单描述。在findFaces方法执行时,会将识别到的人脸信息保存在这个FaceDetector.Face对象数组中
返回值:
- int型值,表示识别检测到的人脸数量
在布局文件中,按钮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);
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新
如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)
最后
说一千道一万,不如自己去行动。要想在移动互联网的下半场是自己占有一席之地,那就得从现在开始,从今天开始,马上严格要求自己,既重视业务实现能力,也重视基础和原理。基础夯实好了,高楼才能够平地而起,稳如泰山。
最后为了帮助大家深刻理解Android相关知识点的原理以及面试相关知识,这里放上相关的我搜集整理的24套腾讯、字节跳动、阿里、百度2020-2021面试真题解析,我把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包知识脉络 + 诸多细节。
还有 高级架构技术进阶脑图、Android开发面试专题资料 帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习。
网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。
一个人可以走的很快,但一群人才能走的更远。如果你从事以下工作或对以下感兴趣,欢迎戳这里加入程序员的圈子,让我们一起学习成长!
AI人工智能、Android移动开发、AIGC大模型、C C#、Go语言、Java、Linux运维、云计算、MySQL、PMP、网络安全、Python爬虫、UE5、UI设计、Unity3D、Web前端开发、产品经理、车载开发、大数据、鸿蒙、计算机网络、嵌入式物联网、软件测试、数据结构与算法、音视频开发、Flutter、IOS开发、PHP开发、.NET、安卓逆向、云计算
CODING开源项目:《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》收录**
一个人可以走的很快,但一群人才能走的更远。如果你从事以下工作或对以下感兴趣,欢迎戳这里加入程序员的圈子,让我们一起学习成长!
AI人工智能、Android移动开发、AIGC大模型、C C#、Go语言、Java、Linux运维、云计算、MySQL、PMP、网络安全、Python爬虫、UE5、UI设计、Unity3D、Web前端开发、产品经理、车载开发、大数据、鸿蒙、计算机网络、嵌入式物联网、软件测试、数据结构与算法、音视频开发、Flutter、IOS开发、PHP开发、.NET、安卓逆向、云计算