android 动态人脸识别码,android OpenCV研究之动态人脸识别
随着直播渐渐的火起来,像抱着直播大腿的其他功能也渐渐的火起来了,比如说人脸识别。说起人脸识别用处甚广,比如说有以这个功能为核心的app:美颜相机、美图秀秀、SNOW等等,但是美颜相机和美图秀秀是用的国内SDK《Face++》来做的,这个sdk呢好像是他们自己的后台进行识别并不是app本身做识别。这样就跟我们今天要了解的动态识别不是很对路,肯定不能拿到摄像头的一帧画面去调一次接口再接回参数吧,这样性能肯定不行。所以今天就拿SNOW的例子来说,虽然我不知道他是用什么做的,但是我们可以用openCV也能实现。
我们先看看效果图:
实现步骤如下:
2、然后新建个项目我这里以studio里为基准,在main目录里面新建jniLibs文件夹,为什么叫jniLibs呢,因为这是调用c库的默认文件夹命名,当然你也可以命名其他的,但是需要在build里面指定这个文件夹。好了,打开我们刚才下载的文件,然后一次打开sdk\native\libs,最后把libs目录里面的所有文件夹拷贝到jniLibs里面去。请看图:
3、加好jniLibs之后呢还需要导入一个module,在studio里面点击file->new->import module->导入module目录是刚才下载的sdk\java这个目录。请看图:
4、导入之后呢右键项目打开open module setting选项,在app选项里点击Dependencies这个,然后点击最右边的+号把刚刚导入的module加进去。请看图:
5、现在开始写代码了,这里我把需要写的代码文件会一一贴出来,下面请看图:
首先是MainActivity的代码:
package com.wyw.facedemo;
import android.content.Context;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.View;
import android.view.WindowManager;
import android.widget.Button;
import android.widget.RelativeLayout;
import org.opencv.android.CameraBridgeViewBase;
import org.opencv.android.JavaCameraView;
import org.opencv.android.OpenCVLoader;
import org.opencv.core.Core;
import org.opencv.core.CvType;
import org.opencv.core.Mat;
import org.opencv.core.MatOfRect;
import org.opencv.core.Rect;
import org.opencv.core.Scalar;
import org.opencv.core.Size;
import org.opencv.objdetect.CascadeClassifier;
import java.io.File;
import java.io.FileOutputStream;
import java.io.InputStream;
public class MainActivity extends AppCompatActivity implements CameraBridgeViewBase.CvCameraViewListener {
private CameraBridgeViewBase openCvCameraView;
private CascadeClassifier cascadeClassifier;
//图像人脸小于高度的多少就不检测
private int absoluteFaceSize;
//临时图像对象
private Mat matLin;
//最终图像对象
private Mat mat;
//前置摄像头
public static int CAMERA_FRONT = 0;
//后置摄像头
public static int CAMERA_BACK = 1;
private int camera_scene = CAMERA_BACK;
private void initializeOpenCVDependencies() {
try {
// Copy the resource into a temp file so OpenCV can load it
InputStream is = getResources().openRawResource(R.raw.lbpcascade_frontalface);
File cascadeDir = getDir("cascade", Context.MODE_PRIVATE);
File mCascadeFile = new File(cascadeDir, "lbpcascade_frontalface.xml");
FileOutputStream os = new FileOutputStream(mCascadeFile);
byte[] buffer = new byte[4096];
int bytesRead;
while ((bytesRead = is.read(buffer)) != -1) {
os.write(buffer, 0, bytesRead);
}
is.close();
os.close();
// Load the cascade classifier
cascadeClassifier = new CascadeClassifier(mCascadeFile.getAbsolutePath());
} catch (Exception e) {
Log.e("OpenCVActivity", "Error loading cascade", e);
}
// And we are ready to go
openCvCameraView.enableView();
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
getWindow().addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON);
setContentView(R.layout.activity_main);
final RelativeLayout relativeLayout = (RelativeLayout) findViewById(R.id.relative);
openCvCameraView = new JavaCameraView(this, CameraBridgeViewBase.CAMERA_ID_FRONT);
openCvCameraView.setCvCameraViewListener(this);
final Button button = new Button(MainActivity.this);
button.setText("切换摄像头");
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if (camera_scene == CAMERA_FRONT) {//如果是前置摄像头就切换成后置
relativeLayout.removeAllViews();
openCvCameraView.disableView();
openCvCameraView = null;
cascadeClassifier = null;
openCvCameraView = new JavaCameraView(MainActivity.this, CameraBridgeViewBase.CAMERA_ID_BACK);
openCvCameraView.setCvCameraViewListener(MainActivity.this);
openCvCameraView.setCameraIndex(CameraBridgeViewBase.CAMERA_ID_BACK);//后置摄像头
camera_scene = CAMERA_BACK;
relativeLayout.addView(openCvCameraView);
relativeLayout.addView(button);
initializeOpenCVDependencies();
} else {
relativeLayout.removeAllViews();
openCvCameraView.disableView();
openCvCameraView = null;
cascadeClassifier = null;
openCvCameraView = new JavaCameraView(MainActivity.this, CameraBridgeViewBase.CAMERA_ID_FRONT);
openCvCameraView.setCvCameraViewListener(MainActivity.this);
openCvCameraView.setCameraIndex(CameraBridgeViewBase.CAMERA_ID_FRONT);//前置摄像头
camera_scene = CAMERA_FRONT;
relativeLayout.addView(openCvCameraView);
relativeLayout.addView(button);
initializeOpenCVDependencies();
}
}
});
relativeLayout.addView(openCvCameraView);
relativeLayout.addView(button);
if (camera_scene == CAMERA_FRONT) {
openCvCameraView.setCameraIndex(CameraBridgeViewBase.CAMERA_ID_FRONT);//前置摄像头
} else if (camera_scene == CAMERA_BACK) {
openCvCameraView.setCameraIndex(CameraBridgeViewBase.CAMERA_ID_BACK);//后置摄像头
}
}
@Override
public void onCameraViewStarted(int width, int height) {
matLin = new Mat(height, width, CvType.CV_8UC4);//临时图像
// 人脸小于高度的百分之30就不检测
absoluteFaceSize = (int) (height * 0.3);
}
@Override
public void onCameraViewStopped() {
}
@Override
public Mat onCameraFrame(Mat aInputFrame) {
//转置函数,将图像翻转(顺时针90度)
Core.transpose(aInputFrame, matLin);
if (camera_scene == CAMERA_FRONT) {//前置摄像头
//转置函数,将图像翻转(对换)
Core.flip(matLin, aInputFrame, 1);
//转置函数,将图像顺时针顺转(对换)
Core.flip(aInputFrame, matLin, 0);
mat = matLin;
} else if (camera_scene == CAMERA_BACK) {//后置摄像头
//转置函数,将图像翻转(对换)
Core.flip(matLin, aInputFrame, 1);
mat = aInputFrame;
}
MatOfRect faces = new MatOfRect();
Log.i("123456", "absoluteFaceSize = " + absoluteFaceSize);
// Use the classifier to detect faces
if (cascadeClassifier != null) {
cascadeClassifier.detectMultiScale(mat, faces, 1.1, 1, 1,
new Size(absoluteFaceSize, absoluteFaceSize), new Size());
}
// 检测出多少个
Rect[] facesArray = faces.toArray();
for (int i = 0; i < facesArray.length; i++) {
Log.i("123456", "facesArray[i].tl()坐上坐标 == " + facesArray[i].tl() + " facesArray[i].br() == 右下坐标" + facesArray[i].br());
Core.rectangle(mat, facesArray[i].tl(), facesArray[i].br(), new Scalar(0, 255, 0, 255), 3);
}
return mat;
}
@Override
public void onResume() {
super.onResume();
if (!OpenCVLoader.initDebug()) {
Log.e("log_wons", "OpenCV init error");
// Handle initialization error
}
initializeOpenCVDependencies();
//OpenCVLoader.initAsync(OpenCVLoader.OPENCV_VERSION_2_4_6, this, mLoaderCallback);
}
}
然后是layout的xml代码:
现在是raw文件夹里面的xml(这个xml是图片解析出来进行对比校验人脸的模型库)由于这个文件有一千多行就不贴了,如有需要请去下载本demo查看!当然也可以去你下载的openCV的sdk里面拿,目录是\samples\face-detection\res\raw。请看图:
最后就是AndroidManifest文件了:
做到这一步就赶紧把你的代码运行起来吧!!本篇博客就到这里,如果有有疑问的欢迎留言讨论。同时希望大家多多关注我的博客,多多支持我。
android 动态人脸识别码,android OpenCV研究之动态人脸识别相关推荐
- android 动态人脸识别码,Android开发中人脸识别(静态)
知道没有妹纸,你们是不会看的.先放效果图 最近,项目中需要用到人脸识别,苦于无奈,各种百度,google有关Android开发中人脸识别的内容,最终发现Android官方自带的FaceDetector ...
- 【Android App】利用自带的人脸检测器和OpenCV检测人脸讲解及实战(附源码和演示 超详细)
需要源码请点赞关注收藏后评论区留言私信~~~~ 一.利用人脸检测器识别人脸 对于简单的人脸识别操作,Android已经提供了专门的识别工具,名叫人脸检测器FaceDetector,部分常用方法如下 F ...
- Android apk动态加载机制的研究(二):资源加载和activity生命周期管理
转载请注明出处:http://blog.csdn.net/singwhatiwanna/article/details/23387079 (来自singwhatiwanna的csdn博客) 前言 为了 ...
- 基于Java的Android OpenCV安装配置及人脸识别示例
1. OpenCV安装配置 1.1 开发环境 Android Studio 3.5.2 Opencv-4.3.0-android-sdk 1.2 安装OpenCV SDK opencv的官网中下载an ...
- Android apk动态加载机制的研究
转载请注明出处:http://blog.csdn.net/singwhatiwanna/article/details/22597587 (来自singwhatiwanna的csdn博客) 背景 问题 ...
- Android研究学习动态壁纸
1. 创建一个类集成 WallpaperService public class MyWallpaperService extends WallpaperService {@Overridepubli ...
- (转载)Android GradientDrawable(shape标签定义) 静态使用和动态使用(圆角,渐变实现)
最近被吐槽界面太丑,还是很尴尬的,全公司就一个UI设计师,所以很多事情还是不忍直视,一个同事问我,背景可不可以使用渐变的感觉,然后我就有种突然感觉眼前一亮的感觉.还真的没有做过这方面的东西,单纯使用渐 ...
- 常见android app加固厂商脱壳方法研究
这篇文章主要介绍了常见android app加固厂商脱壳方法研究,需要的朋友可以参考下 目录简述(脱壳前学习的知识.壳的历史.脱壳方法) 第一代壳 第二代壳 第三代壳 第N代壳 简述Apk文件结构De ...
- Android一键生成包含.dex的Jar及动态加载方案
Android一键生成包含.dex的Jar及动态加载方案 背景:谈到动态加载相信很多小伙伴都会想到 热更新 及 动态加载dex 的技术,最近也因为项目重构的需求,折腾了下这方面的技术点,以前研究过但时 ...
最新文章
- 交叉熵损失函数分类_交叉熵损失函数
- pku 1573 Robot Motion 第一周训练——模拟
- 程序员食品营养(1)-面包基础
- nodejs之日志管理
- python运算符讲解_python运算符讲解
- AbstractEndpoint 和 ProtocolHandler
- 【华为云 ModelArts-Lab AI实战营】第二期
- fgets,cin. getline被跳过
- Web专家周末聚会都会说点啥
- 应急卫星通信系统项目建设要求
- 生活小妙招:办公室如何查看自己电脑已连接的无线网密码,记得收藏
- vue 中的slot属性(插槽)的使用
- 【YOLO】YOLO简介
- 美团后台开发实习生面试经验
- Windows独享主机如何查看服务器系统日志?
- Markdown 图片自动上传
- CSS固定格式-小垒
- Spark SQL 外部数据源
- svg loading效果(上百个)
- laravel6.x + laravel-admin v1
热门文章
- 【JavaScript总结】JavaScript发展与学习内容
- GeneratorSqlMapCustom(mybatis逆向工程)
- OFRecord 数据格式
- 视频动作定位的分层自关注网络:ICCV2019论文解析
- 2021年大数据Hadoop(二十五):YARN通俗介绍和基本架构
- Linux环境变量说明与配置
- cc.tween 的call()不执行
- [JavaScript] 日期时间戳的使用与计算
- Error:(17, 0) SDK location not found. Define location with sdk.dir in the local.properties file or w
- ABAP 程序间的调用