草庐IT

NDK OpenCV人脸定位

sziitjin 2024-04-08 原文

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人脸定位

有关NDK OpenCV人脸定位的更多相关文章

  1. 「Python|Selenium|场景案例」如何定位iframe中的元素? - 2

    本文主要介绍在使用Selenium进行自动化测试或者任务时,对于使用了iframe的页面,如何定位iframe中的元素文章目录场景描述解决方案具体代码场景描述当我们在使用Selenium进行自动化测试的时候,可能会遇到一些界面或者窗体是使用HTML的iframe标签进行承载的。对于iframe中的标签,如果直接查找是无法找到的,会抛出没有找到元素的异常。比如近在咫尺的例子就是,CSDN的登录窗体就是使用的iframe,大家可以尝试通过F12开发者模式查看到的tag_name,class_name,id或者xpath来定位中的页面元素,会抛出NoSuchElementException异常。解决

  2. 最新版人脸识别小程序 图片识别 生成二维码签到 地图上选点进行位置签到 计算签到距离 课程会议活动打卡日常考勤 上课签到打卡考勤口令签到 - 2

    技术选型1,前端小程序原生MINA框架cssJavaScriptWxml2,管理后台云开发Cms内容管理系统web网页3,数据后台小程序云开发云函数云开发数据库(基于MongoDB)云存储4,人脸识别算法基于百度智能云实现人脸识别一,用户端效果图预览老规矩我们先来看效果图,如果效果图符合你的需求,就继续往下看,如果不符合你的需求,可以跳过。1-1,登录注册页可以看到登录页有注册入口,注册页如下我们的注册,需要管理员审核,审核通过后才可以正常登录使用小程序1-2,个人中心页登录成功以后,我们会进入个人中心页我们在个人中心页可以注册人脸,因为我们做人脸识别签到,需要先注册人脸才可以进行人脸比对,进

  3. 基于Python的人脸识别课堂系统(毕设)——附录上 - 2

    本文章承接《基于Python的人脸识别课堂考勤系统(毕设)》,填坑上篇文章遗留的代码部分。因为项目分的模块比较多,再加上本人能力有限,所以代码过于臃肿还存在许多优化的地方。同样本篇文章也仅适用于小白,零基础人群。PS:每个文件之中代码都已经区分开来,可以对照左侧目录部分实现快速预览!    由于代码过于多我这里分成上,下两个部分来发布吧!一、主文件importosimportsysimportrandomimportpymysqlimportcv2importnumpyasnpfrommathimportpifrommatplotlibimportpyplotaspltfromPILimpor

  4. 打通源码,高效定位代码问题|云效工程师指北 - 2

    大家好,我叫胡飞虎,花名虎仔,目前负责云效旗下产品Codeup代码托管的设计与开发。代码作为企业最核心的数据资产,除了被构建、部署之外还有更大的价值。为了帮助企业和团队挖掘更多源代码价值以赋能日常代码研发、运维等工作,云效代码团队在大数据和智能化方向进行了一系列的探索和实践(例如代码搜索与推荐),本文主要介绍我们如何通过直接打通源代码来提高研发与运维效率。随着微服务架构的流行,一个业务流程需要多个微服务共同完成。一旦出现问题,运维人员在面对数量多、调用链路复杂的情况下,很难快速锁定导致问题发生的罪魁祸首:代码。为了提高排查效率,目前常见的解决方案是:链路跟踪+日志分析工具相结合。即通过链路跟踪

  5. ruby - 在 Mac 上定位 "irbrc"文件 - 2

    我看到很多很酷的东西可以添加到我的Ruby控制台中。例如,一个好的列表是“My.irbrcforconsole/irb”。我用谷歌搜索,但我只找到了网络日志,上面写着人们在他们的.irbrc中添加了什么gem。没有人说在哪里可以找到它。我找不到“irbrc”。我打开了我的主文件夹,如果我输入IRB,它会转到Ruby控制台,但我找不到这个文件。谁能帮我找到它? 最佳答案 这是一个irbrc点文件,因此您需要在您的主目录中ls-a才能找到它。如果它不在那里,只需创建一个.irbrc文件。我的很简单,但这就是我的内容:require'ru

  6. ruby-on-rails - capybara :查找(元素)使用选择器来定位复杂的属性名称 - 2

    使用cucumber和capybara测试Rails应用。假设我无法更改标记,我可以使用capybara在充满类似td和select的页面中选择以下选择吗?LanguagesCommunication这似乎失败了(我假设是因为嵌套的“[”和“]”)。find("select[name=attributes[ruby][category]]")转义也不行。想法? 最佳答案 您可以尝试find('select',:name=>'attributes[ruby][category]')或find_field('attributes[rub

  7. 基于对象属性定位的 Ruby 数组方法? - 2

    假设我有一个Ruby类,Flight。Flight上有一个attr_accessor:key。如果有一个此类的实例数组:flights=[flight1,flight2,flight3],我有一个“目标键”,比如说“2jf345”,我想根据它找到一个航类键,来自该数组-我应该使用哪种代码?这是我要使用的代码:航类[flights.map{|s|s.key}.index(target_key)]但是对于Ruby,似乎应该有更简单的方法。此外,上面的代码为我返回了一个错误-`[]':noimplicitconversionfromniltointeger(TypeError)。我认为这意味

  8. ruby - 在 Ruby 中将元素重新定位到数组的前面 - 2

    即使来自javascript,这对我来说也很糟糕:irb>>a=['a','b','c']=>["a","b","c"]>>a.unshift(a.delete('c'))=>["c","a","b"]有没有更清晰的方法将元素放在数组的前面?编辑我的实际代码:if@admin_users.include?(current_user)@admin_users.unshift(@admin_users.delete(current_user))end 最佳答案 也许这对你来说更好看:a.insert(0,a.delete('c'))

  9. ruby-on-rails - Ruby 地理定位 Gem/插件 - 2

    关闭。这个问题不符合StackOverflowguidelines.它目前不接受答案。要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于StackOverflow来说是偏离主题的,因为它们往往会吸引自以为是的答案和垃圾邮件。相反,describetheproblem以及迄今为止为解决该问题所做的工作。关闭9年前。Improvethisquestion什么可用的(最好的)基于ruby​​IP的地理定位gem/插件?它们在功能、性能和易用性方面如何相互比较(例如,它们是否与网络服务交互,或需要单独的数据库等)?希望有使用过的friend可以分享一下经验和建议。

  10. ruby-on-rails - 如何使用 capistrano deploy 定位特定的提交 SHA - 2

    我想知道如何使用Capistrano在Git中针对特定的提交SHA进行部署?应该是这样的capdeploy--version=经过大量搜索似乎无法找到这个问题的答案。 最佳答案 对于Capistrano2.9到3.0:cap-Srevision=80655da8d80aaaf92ce5357e7828dc09adb00993deploy对于旧版本的Capistrano,您可以通过执行以下操作来部署特定的gitcommit/tree/branch/tag:cap-sbranch=80655da8d80aaaf92ce5357e7828

随机推荐