Android OpenCV实现人脸检测(二)完成预览画面显示

Android OpenCV 人脸识别系列文章:

Android Studio 集成OpenCV
Android OpenCV实现人脸检测(一)完成人脸检测功能

代码链接:

Android OpenCV 实现人脸检测
在这里插入图片描述

注意:如果你之前下载过本代码,安装运行的时候预览画面停下来了,并且之前也安装过这个应用,请先把应用卸载掉,然后点击 Android Studio的 Build -> Clean Project -> 运行,安装后即可运行。
又或者是:你下载了本代码,然后修改了 JNI 层的代码 然后运行卡到预览画面,那么也请你按上述步骤做。

在上面两篇文章的基础上,我们实现了人脸检测,并将检测出的人脸变为 24 * 24 大小的灰度图存放在 /data/data/com.example.myopencv/cache/face_img_24x24/ 这个目录下。
在这里插入图片描述
接下来,我们需要将画面渲染到预览Surface。

1.引入 libjnigraphics.so 和 libandroid.so 这两个系统的库

其中 libjnigraphics.so 是 Java 层传 Bitmap 到 JNI 层需要用到的库。由于这两个是系统自带的库,所以我们只需要在 target_link_libraries 引入即可。

target_link_libraries(
        myopencv
        opencv_java4
        ${log-lib}

        android
        jnigraphics
#        千万不要写成下面这种形式,下面这种写法不能引用系统库
#        ${android}
#        ${jnigraphics} # Java 层传 Bitmap 到 JNI 层需要用到的库
)

在这里插入图片描述

2.实现 Java 层传递 Surface 到 JNI 层
native void setSurface(Surface surface);

@Override
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
    //设置surface 用于显示
    setSurface(holder.getSurface());
}
extern "C"
JNIEXPORT void JNICALL
Java_com_example_myopencv_MainActivity_setSurface(JNIEnv *env, jobject thiz, jobject surface) {

    // 如果window已经存在,就释放掉
    if (window){
        ANativeWindow_release(window);
        window = NULL;
    }

    window = ANativeWindow_fromSurface(env, surface);
}
3.JNI层将人脸渲染到预览画面

修改 JNI 的 postData 方法:

for (Rect face : faces) {
        //要将下面几行注释掉,否则预览画面只会停在显示一帧
//        sprintf(picture_name, "/data/data/com.example.myopencv/cache/face_img_24x24/%d.png", index++);
//        Mat face_rect;
//        face_rect = gray_img(face).clone();
//        resize(face_rect, face_rect, Size(24, 24));
//        imwrite(picture_name, face_rect);

        // 在原始的预览画面中画人脸区域的框框
        rectangle(src, face,Scalar(0, 0, 255));
    }

    // 将人脸渲染到预览画面
    if (window){

        do {
            ANativeWindow_setBuffersGeometry(window, src.cols, src.rows, WINDOW_FORMAT_RGBA_8888);
            ANativeWindow_Buffer buffer;
			
            if (ANativeWindow_lock(window, &buffer, 0)){
                ANativeWindow_release(window);
                window = NULL;
            }

            int srcLineSize = src.cols * 4;
            int dstLineSize = buffer.stride * 4;
            uint8_t *dstData = static_cast<uint8_t *> (buffer.bits);
            uint8_t *srcData = src.data;
            for (int i = 0; i < buffer.height; ++i) {
                memcpy(dstData + dstLineSize * i, srcData + srcLineSize * i, srcLineSize);
            }
            // 如果没有下面这行,则画面只会显示一帧
            ANativeWindow_unlockAndPost(window);
        } while (0);
    }

    src.release();
    gray_img.release();

编译运行后,预览画面显示出来
在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值