NDK OpenCV人脸定位

news2024/12/25 12:36:01

NDK系列之OpenCV人脸定位技术实战,本节主要是通过OpenCV C++库,实现识别人脸定位,并对识别到的人脸画面增加红框显示。

实现效果:

实现逻辑:

1.初始化CameraX,绑定图片分析器ImageAnalysis,监听相机数据;

2.加载OpenCV提供的人脸识别训练数据lbpcascade_frontalface到本地;

3.初始化人脸跟踪中转站FaceTracker,将人脸识别训练数据路径传递到Native层;

4.Native读取人脸识别训练数据,创建人脸检测跟踪器Ptr<DetectionBasedTracker> tracker;

5.通过中转站FaceTracker,调用Native层tracker开启人脸跟踪;

6.通过中转站FaceTracker,实例化Native层播放窗口ANativeWindow,关联surfaceView;

7.获取相机数据,传递Native层,人脸定位,绘制人脸框,渲染画面到屏幕。

本节主要内容:

1.OpenCV库导入;

2.Java层CameraX使用;

3.Native层识别人脸和画面渲染;

源码:

NdkHeadTest: NDK OpenCV人脸定位

一、OpenCV库导入

1)复制OpenCV源文件到cpp目录下,动态库文件复制到jniLibs目录下:

2)在CMakeLists文件中,导入源文件和库文件

二、Java层CameraX使用

1)初始化CameraX,绑定图片分析器ImageAnalysis,监听相机数据;

private void initCamera() {
	/**
	 *  CameraX
	 */
	cameraProviderFuture = ProcessCameraProvider.getInstance(this);
	cameraProviderFuture.addListener(() -> {
		try {
			ProcessCameraProvider cameraProvider = cameraProviderFuture.get();
			bindAnalysis(cameraProvider);
		} catch (Exception e) {
			e.printStackTrace();
		}

	}, ContextCompat.getMainExecutor(this));
}

private void bindAnalysis(ProcessCameraProvider cameraProvider) {
	//STRATEGY_KEEP_ONLY_LATEST :非阻塞模式,每次获得最新帧
	//STRATEGY_BLOCK_PRODUCER : 阻塞模式,处理不及时会导致降低帧率
	//图片分析:得到摄像头图像数据
	ImageAnalysis imageAnalysis =
			new ImageAnalysis.Builder()
					.setTargetResolution(new Size(640, 480))
					.setBackpressureStrategy(ImageAnalysis.STRATEGY_KEEP_ONLY_LATEST)
					.build();
	imageAnalysis.setAnalyzer(ContextCompat.getMainExecutor(this), this);
	cameraProvider.unbindAll();
	//绑定生命周期
	cameraProvider.bindToLifecycle(this,
			CameraSelector.DEFAULT_BACK_CAMERA, imageAnalysis);
}

2)相机数据会通过ImageAnalysis.Analyzer接口回调到analyze(@NonNull ImageProxy image)

@Override
public void analyze(@NonNull ImageProxy image) {
	byte[] bytes = Utils.getDataFromImage(image);
	// 定位人脸,并且显示摄像头的图像
	faceTracker.detect(bytes, image.getWidth(), image.getHeight(), image.getImageInfo().getRotationDegrees());
	image.close();
}

三、Native层识别人脸和画面渲染

1)加载OpenCV提供的人脸识别训练数据lbpcascade_frontalface到本地;

String path = Utils.copyAsset2Dir(this, "lbpcascade_frontalface.xml");

2)初始化人脸跟踪中转站FaceTracker,将人脸识别训练数据路径传递到Native层;

faceTracker = new FaceTracker(path);

public FaceTracker(String model) {
	mNativeObj = nativeCreateObject(model);
}

private static native long nativeCreateObject(String model);
	

 Native层接收到人脸识别训练数据路径,初始化FaceTracker.cpp

extern "C"
JNIEXPORT jlong JNICALL
Java_com_ndk_head_FaceTracker_nativeCreateObject(JNIEnv *env, jclass clazz, jstring model_) {
    // 转换人脸训练模型数据为char *
    const char *model = env->GetStringUTFChars(model_, 0);
    FaceTracker *tracker = new FaceTracker(model);
    env->ReleaseStringUTFChars(model_, model);
    // 返回tracker地址给Java层
    return (jlong) tracker;
}	

3)Native读取人脸识别训练数据,创建人脸检测跟踪器Ptr<DetectionBasedTracker> tracker;

FaceTracker::FaceTracker(const char *model) {
    // 初始化互斥锁
    pthread_mutex_init(&mutex, 0);
    // 创建检测器适配器
    Ptr<CascadeDetectorAdapter> mainDetector = makePtr<CascadeDetectorAdapter>(
            makePtr<CascadeClassifier>(model));
    Ptr<CascadeDetectorAdapter> trackingDetector = makePtr<CascadeDetectorAdapter>(
            makePtr<CascadeClassifier>(model));
    //跟踪器
    DetectionBasedTracker::Parameters DetectorParams;
    tracker = makePtr<DetectionBasedTracker>(DetectionBasedTracker(mainDetector, trackingDetector,
                                                                   DetectorParams));
}

4)通过中转站FaceTracker,调用Native层tracker开启人脸跟踪;

faceTracker.start();

public void start() {
	nativeStart(mNativeObj);
}
	
private static native void nativeStart(long thiz);	

Native层开启人脸跟踪 

extern "C"
JNIEXPORT void JNICALL
Java_com_ndk_head_FaceTracker_nativeStart(JNIEnv *env, jclass clazz, jlong thiz) {
    if (thiz != 0) {
        FaceTracker *tracker = reinterpret_cast<FaceTracker *>(thiz);
        tracker->tracker->run();
    }
}

5)通过中转站FaceTracker,实例化Native层播放窗口ANativeWindow,关联surfaceView;

@Override
public void surfaceChanged(@NonNull SurfaceHolder holder, int format, int width, int height) {
	if (faceTracker != null)
		faceTracker.setSurface(holder.getSurface());
}

public void setSurface(Surface surface) {
	nativeSetSurface(mNativeObj, surface);
}

private static native void nativeSetSurface(long thiz, Surface surface);

Native层实例化ANativeWindow,关联surfaceView 

extern "C"
JNIEXPORT void JNICALL
Java_com_ndk_head_FaceTracker_nativeSetSurface(JNIEnv *env, jclass clazz, jlong thiz,
                                               jobject surface) {
    if (thiz != 0) {
        FaceTracker *tracker = reinterpret_cast<FaceTracker *>(thiz);
        if (!surface) {
            tracker->setANativeWindow(0);
            return;
        }
        tracker->setANativeWindow(ANativeWindow_fromSurface(env, surface));
    }
}

6)获取相机数据,传递Native层,人脸定位,绘制人脸框,渲染画面到屏幕。

@Override
public void analyze(@NonNull ImageProxy image) {
	byte[] bytes = Utils.getDataFromImage(image);
	// 定位人脸,并且显示摄像头的图像
	faceTracker.detect(bytes, image.getWidth(), image.getHeight(), image.getImageInfo().getRotationDegrees());
	image.close();
}

public void detect(byte[] inputImage, int width, int height, int rotationDegrees) {
	nativeDetect(mNativeObj, inputImage, width, height, rotationDegrees);
}

private static native void nativeDetect(long thiz, byte[] inputImage, int width, int height, int rotationDegrees);

Native层识别人脸,绘制人脸红框 

extern "C"
JNIEXPORT void JNICALL
Java_com_ndk_head_FaceTracker_nativeDetect(JNIEnv *env, jclass clazz, jlong thiz,
                                           jbyteArray inputImage_, jint width, jint height,
                                           jint rotationDegrees) {
    if (thiz == 0) {
        return;
    }
    FaceTracker *tracker = reinterpret_cast<FaceTracker *>(thiz);
    // 将图片数据转化为jbyte
    jbyte *inputImage = env->GetByteArrayElements(inputImage_, 0);
    // 根据I420宽高,设置Mat(OpenCV支持的图片格式)的宽高,并赋值 src
    Mat src(height * 3 / 2, width, CV_8UC1, inputImage);
    // YUV转为RGBA
    cvtColor(src, src, CV_YUV2RGBA_I420);
    // 旋转图片
    if (rotationDegrees == 90) {
        rotate(src, src, ROTATE_90_CLOCKWISE);
    } else if (rotationDegrees == 270) {
        rotate(src, src, ROTATE_90_COUNTERCLOCKWISE);
    }
    Mat gray; // 存储降噪后的图片(灰度图)
    // 灰度化
    cvtColor(src, gray, CV_RGBA2GRAY);
    // 增强对比度
    equalizeHist(gray, gray);
    // 人脸定位
    tracker->tracker->process(gray);

    std::vector<Rect> faces; // 人脸集合
    tracker->tracker->getObjects(faces);

    for (Rect face:faces) {
        // 找到人脸,画红色矩形框
        rectangle(src, face, Scalar(255, 0, 0));
    }

    tracker->draw(src);
    env->ReleaseByteArrayElements(inputImage_, inputImage, 0);
}

将最终定位完成的图片绘制到屏幕

void FaceTracker::draw(Mat img) {
    pthread_mutex_lock(&mutex);
    do {
        if (!window) {
            break;
        }
        // 设置window格式
        ANativeWindow_setBuffersGeometry(window, img.cols, img.rows,
                                         WINDOW_FORMAT_RGBA_8888);
        // 把需要显示的数据设置给buffer
        ANativeWindow_Buffer buffer;
        if (ANativeWindow_lock(window, &buffer, 0)) {
            ANATIVEWINDOW_RELEASE(window);
            break;
        }
        // 把视频数据刷新到buffer中
        uint8_t *dstData = static_cast<uint8_t *>(buffer.bits);
        int dstlineSize = buffer.stride * 4;
        // 视频图形rgba数据
        uint8_t *srcData = img.data;
        int srclineSize = img.cols * 4;
        // 一行一行的拷贝
        for (int i = 0; i < buffer.height; ++i) {
            memcpy(dstData + i * dstlineSize, srcData + i * srclineSize, srclineSize);
        }
        // 提交渲染
        ANativeWindow_unlockAndPost(window);
    } while (0);
    pthread_mutex_unlock(&mutex);
}

至此,OpenCV人脸识别定位技术项目已完成;同时人眼识别定位等相关实现也是雷同的,都可以通过OpenCV实现,后续会通过OpenCV与OpenGL实现大眼萌特效。

源码:

NdkHeadTest: NDK OpenCV人脸定位

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/449808.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

7.队列算法

算法&#xff1a;队列算法 队列是一种抽象的数据结构&#xff0c;有点类似于Stacks。与堆栈不同&#xff0c;队列的两端都是开放的。一端始终用于插入数据(入队)&#xff0c;另一端用于删除数据(出队)。队列遵循先进先出方法&#xff0c;即首先访问先存储的数据项。 一个真实的…

【C++初阶】类与对象(上)

一.什么是类&#xff0c;什么是对象 我们可以形象的把类比作是一个房子的设计图纸&#xff0c;而对象就是根据设计图纸设计出来的房子。 由设计图纸到房子的过程&#xff0c;我们称之为类的实例化。 C兼容C的&#xff0c;所以C中的结构体在C中也能用&#xff0c;但是C把结构体升…

rust教程 第一章 —— 初识rust

文章目录 前言一、Rust简介二、安装Rust编译器三、第一个Rust程序四、 IDE环境五、初识包管理六、总结 前言 本系列教程目录可查看这里&#xff1a;Rust教程目录 近些年来不断有新的语言崛起&#xff0c;比如当下非常火的go语言&#xff0c;不过相比于C&#xff0c;go语言确实…

C++类和对象 (3)

类和对象 1. 类的6个默认成员函数2. 构造函数2.1. 概念&#xff08;问题提出&#xff09;2.2. 特性 3.析构函数3.1. 概念3.2.特性 1. 类的6个默认成员函数 如果一个类中什么成员都没有&#xff0c;简称为空类。 空类中真的什么都没有吗&#xff1f;并不是&#xff0c;任何类在…

使用OpenFeign实现接口访问

1. 引言 在微服务横行的年代&#xff0c;后端根据业务的不一样分成了很多单独运行的服务&#xff0c;比如在物联网中&#xff0c;根据业务拆分为定时服务、设备控制等服务。当前端想控制设备时&#xff0c;其请求首先到其对应的后端服务&#xff0c;后端服务再调用设备控制服务…

Vue+Vant封装通用模态框单选框组件

前言 我们知道&#xff0c;在vant组件中提供的组件往往是比较基础的&#xff0c;能够满足基本需求。但是我们想实现ui设计的一些比较丰富效果的组件&#xff0c;需要自己去实现&#xff0c;且当项目中多次用到的时候&#xff0c;我们将以组件化的思想将其封装起来&#xff0c;…

Node.js -- 使用Express写接口

1.创建基本的服务器 //导入express const express require(express) //创建服务器实例 const app express() //调用app.listen方法&#xff0c;指定端口号并启动web服务器 app.listen(80,function(){console.log(Express server running at http://127.0.0.1) })2. 创建API路…

路由交换综合实验

拓扑结构&#xff1a; 要求 1、R6为网络运营商&#xff08;ISP&#xff09;&#xff0c;接口IP地址均为公有地址&#xff1b;该设备只能配置IP地址&#xff0c;之后不能在对其进行任何配置&#xff1b; 2、R1~R5为局域网&#xff0c;私有IP地址192.168.1.0/24&#xff0c;请合…

真题详解(UML图)-软件设计(五十五)

真题详解&#xff08;计算机知识&#xff09;-软件设计&#xff08;五十四)https://blog.csdn.net/ke1ying/article/details/130278265 组织域名&#xff1a; com商业组织 edu教育组织 gov政府组织 net主要网络支持中心 mil军事部门 Int国际组织 2、时间复杂度 O&#…

写一个自己的命令行解释器

写一个自己的命令行解释器 当我点开xshell运行服务器的时候bash就被加载到了内存中&#xff0c;此后我在bash上执行的所有程序都是作为bash的子进程。在bash这个进程内创建子进程&#xff0c;并让子进程去执行全新的代码&#xff0c;这不就是程序替换吗&#xff1f; 所以我们…

腾讯云4核8g服务器支持多少人在线使用?

腾讯云轻量4核8G12M轻量应用服务器支持多少人同时在线&#xff1f;通用型-4核8G-180G-2000G&#xff0c;2000GB月流量&#xff0c;系统盘为180GB SSD盘&#xff0c;12M公网带宽&#xff0c;下载速度峰值为1536KB/s&#xff0c;即1.5M/秒&#xff0c;假设网站内页平均大小为60KB…

【Unity入门】17.脚本访问父子结点

【Unity入门】脚本访问父子结点 大家好&#xff0c;我是Lampard~~ 欢迎来到Unity入门系列博客&#xff0c;所学知识来自B站阿发老师~感谢 &#xff08;一&#xff09;父级节点 &#xff08;1&#xff09;访问父级节点 父子关系我们并不陌生&#xff0c;在cocos中常用node:get…

单链表的实现

链表的概念与结构 链表与我们通讯录中的顺序表是不同的&#xff0c;顺序表的空间是连续的&#xff0c;像数组一样可以通过下标访问。而链表是一种物理存储结构上非连续、非顺序的存储结构&#xff0c;数据元素的逻辑顺序是通过链表中的指针链接次序实现的。即&#xff1a;链表…

数据结构笔记:二叉树的遍历与技巧

引言 本篇是最近有遇到过的一个题目&#xff0c;关于二叉树的遍历&#xff0c;正好有一些经验与技巧&#xff0c;所以开一篇帖子记录一下。 二叉树遍历介绍 遍历是数据结构中常见的操作&#xff0c;主要是将所有元素都访问一遍。对于线性结构来说&#xff0c;遍历分为两种&a…

RecyclerView 静态布局实现过程解析:如何构建高性能的列表

作者&#xff1a;maxcion Recyclerview在日常开发中所使用的控件中绝对是顶流一般的存在&#xff0c;想嚼它这个想法一次两次了。在网上也看了很多关于Recyclerview源码解析的文章&#xff0c;大佬们写的都很深刻&#xff0c;但是对于像我们这种储备知识不足的小白读者来说&…

前端实现端到端测试(代码版)

端到端测试框架选取 playwright 、 cypress 、 selenium 对比 cypress使用 下载 cypress npm install cypress --save-dev package.json npm run cypress:open {"scripts": {"cypress:open": "cypress open"} }使用流程 入门官方文档 npm ru…

一本通 3.4.5 最小生成树

1348&#xff1a;【例4-9】城市公交网建设问题 【题目描述】 有一张城市地图&#xff0c;图中的顶点为城市&#xff0c;无向边代表两个城市间的连通关系&#xff0c;边上的权为在这两个城市之间修建高速公路的造价&#xff0c;研究后发现&#xff0c;这个地图有一个特点&…

SQL Server基础 第四章 select定制查询(select中的各种查询筛选条件)

本章主要介绍 select 语句查询数据的基本用法&#xff0c;其中包括查询指定字段信息、条件查询等。 目录 1、比较运算符、逻辑运算符 &#xff08;1&#xff09;查询phone大于500且不是单县的 &#xff08;2&#xff09;查询地址为烟台或者单县但是phone要大于666的 &#…

IMX6ull 之 HelloWorld Led点灯

一 GPIO点灯&#xff0c;嵌入式的helloworld 1 何为GPIO&#xff1f; GPIO只是一个CPU内提供的一种功能外设&#xff0c;CPU外部的I/O引脚会被赋予一种功能&#xff08;GPIO、UART、I2C等&#xff09;&#xff1b;该功能由CPU内外设提供&#xff0c;具体是什么功能由IOMUX…

刷题笔记4-22

目录 1.Java&#xff1a;(a,b)>Math.abs(a-3)-Math.abs(b-3)&#xff1b; 2.字符解释 3.C语言二维数组中a[i]表示ai的地址&#xff0c;而a[i]又可以表示为*&#xff08;ai&#xff09; 4.二维数组在传参时&#xff0c;必须给定列 5.软件开发&#xff1a;观察者模式 6.建…