Android中使用OpenCV

三点:

  1. 集成OpenCV
  2. 使用官方的人脸识别模型写个Demo
  3. 训练库

一、集成OpenCV

OpenCV集成还是很简单的,不需要我们自己去交差编译生成动/静态库,解压后的文件已经包含了动态库。一般套路都是这样,下载库、导入.h和动/静态库、配置CmakeList。详细步骤:

  1. 下载OpenCV官网Android最新版本SDK(这里用的是4.1.0)
  2. AS创建NDK项目,新旧版本AS创建出来的目录结构不太一样,这里把目录贴一下,跟CMake文件中配置文件路径有关系:


    项目目录
  3. 导入.h文件和.so动态库


  4. 在CmakeLists.txt中引入库,修改3处,下面都有注释,这里的路径就是上面贴目录的原因,根据自己AS版本创建出来的目录自行修改。
cmake_minimum_required(VERSION 3.4.1)


add_library(
        native-lib

        SHARED

        native-lib.cpp)

#导入头文件
include_directories(include)

#导入库文件
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -L${CMAKE_SOURCE_DIR}/libs/${ANDROID_ABI}")

find_library(
        log-lib

        log)


target_link_libraries(
        native-lib
        
        #添加opencv_java4
        opencv_java4

        android

        ${log-lib})
  1. build.gradle中添加寻找目录:
    sourceSets {
        main {
            jniLibs.srcDirs = ['src/main/cpp/libs']
        }
    }
  1. 如果编译报错记得clean rebuild,AS有时候抽风

二、使用官方的人脸识别模型写个Demo

  1. 官方的提供的正脸识别模型也在刚才下载的包里包括了,先copy到assets中:


  2. 这里用到两个工具类UtilsCameraHelper就不贴了,Utils 是把文件从assetscopy到外置储存空间的,CameraHelper是打开摄像头的工具类,在onPreviewFrame(..)中把每一帧图片回调给了MainActivity,因为这里并不是把摄像头采集的画面直接放到SurfaceView显示,识别出人脸之后要标记处理,所以把Surface和摄像头数据都传到native层,在native层对图片处理过之后直接写入Surface,也会得到在SurfaceView预览的效果, 代码可以在项目中查看。
  3. 在MainActivity里我们先把对人脸识别模型进行拷贝,然后把SurfaceView和CameraHelper的每一帧数据传递给native层去处理:
public class MainActivity extends AppCompatActivity implements SurfaceHolder.Callback, Camera.PreviewCallback {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        ...

        Utils.copyAssets(this, "lbpcascade_frontalface.xml");
    }

    @Override
    protected void onResume() {
        super.onResume();
        String path = new File(Environment.getExternalStorageDirectory(), "lbpcascade_frontalface.xml").getAbsolutePath();
        cameraHelper.startPreview();
        openCvJni.init(path);

    }

    ...

    @Override
    public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
        openCvJni.setSurface(holder.getSurface());
    }

    @Override
    public void onPreviewFrame(byte[] data, Camera camera) {
        openCvJni.postData(data, CameraHelper.WIDTH, CameraHelper.HEIGHT, cameraId);
    }

    ...
}
  1. OpencvJni是Java层和native层通讯的类,只有native方法声明,用处已注释:
public class OpencvJni {
    static {
        System.loadLibrary("native-lib");
    }

    /**
     * 初始化native层相关逻辑
     * @param path 人脸识别模型路径
     */
    public native void init(String path) ;

    /**
     * 摄像头采集的数据发送到native层
     * @param data 每一帧数据(NV21)
     * @param width 摄像头采集图片的宽
     * @param height 摄像头采集图片的高
     * @param cameraId 摄像头ID(前置、后置)
     */
    public native void postData(byte[] data, int width, int height, int cameraId);

    /**
     * 发送Surface到native,用于把数据后的图像数据直接显示到Surface上
     * @param surface
     */
    public native void setSurface(Surface surface);
}
  1. native层OpenCV初始化相关的代码,这里要创建一个检测器,一个跟踪器。OpenCV的实现并不是每一帧图片都检测,目标检测还是比较耗时耗性能的,检测出目标之后会有跟踪器的工作,这里的具体原理就不瞎扯了,大概是这么个意思。
Java_com_yu_opencvdemo_OpencvJni_init(JNIEnv *env, jobject instance, jstring path_) {
    const char *path = env->GetStringUTFChars(path_, 0);


    //创建检测器
    Ptr<CascadeClassifier> classifier=makePtr<CascadeClassifier>(path);
    Ptr<CascadeDetectorAdapter> mainDetector= makePtr<CascadeDetectorAdapter>(classifier);

    //创建跟踪器
    Ptr<CascadeClassifier> classifier1=makePtr<CascadeClassifier>(path);
    Ptr<CascadeDetectorAdapter> trackingDetector= makePtr<CascadeDetectorAdapter>(classifier1);

    //开始识别,结果在CascadeDetectorAdapter中返回
    DetectionBasedTracker::Parameters DetectorParams;
    tracker= new DetectionBasedTracker(mainDetector, trackingDetector, DetectorParams);
    tracker->run();

    env->ReleaseStringUTFChars(path_, path);
}

这里需要一个CascadeDetectorAdapter,这个类可以从开发包的sample里找到,包括上面的初始化代码也是参考sample里的,有兴趣可以去读相关代码。这个CascadeDetectorAdapter的作用是每一帧检测出目标后的一个回调:

class CascadeDetectorAdapter: public DetectionBasedTracker::IDetector
{
public:
    CascadeDetectorAdapter(cv::Ptr<cv::CascadeClassifier> detector):
            IDetector(),
            Detector(detector)
    {
    }
    void detect(const cv::Mat &Image, std::vector<cv::Rect> &objects)
    {
        Detector->detectMultiScale(Image, objects, scaleFactor, minNeighbours, 0, minObjSize, maxObjSize);
    }

    virtual ~CascadeDetectorAdapter()
    {
    }

private:
    CascadeDetectorAdapter();
    cv::Ptr<cv::CascadeClassifier> Detector;
};

检测到目标后会回调detect,把结果放进objects里,下面我就就可以通过getObjects()获取检测结果

  1. 在postData对应native层实现中调用OpenCV处理图像:
extern "C"
JNIEXPORT void JNICALL
Java_com_yu_opencvdemo_OpencvJni_postData(JNIEnv *env, jobject instance, jbyteArray data_,
                                          jint width, jint height, jint cameraId) {
    jbyte *data = env->GetByteArrayElements(data_, NULL);

    // 数据的行数也就是数据高度,因为数据类型是NV21,所以为Y+U+V的高度, 也就是height + height/4 + height/4
    Mat src(height*3/2, width, CV_8UC1, data);

    // 转RGB
    cvtColor(src, src, COLOR_YUV2RGBA_NV21);
    if (cameraId == 1) {// 前置摄像头
        //逆时针旋转90度
        rotate(src, src, ROTATE_90_COUNTERCLOCKWISE);
        //1:水平翻转   0:垂直翻转
        flip(src, src, 1);
    } else {
        //顺时针旋转90度
        rotate(src, src, ROTATE_90_CLOCKWISE);

    }
    Mat gray;
    //灰度化
    cvtColor(src, gray, COLOR_RGBA2GRAY);
    //二值化
    equalizeHist(gray, gray);

    std::vector<Rect> faces;
    //检测图片
    tracker->process(gray);
    //获取CascadeDetectorAdapter中的检测结果
    tracker->getObjects(faces);
    //画出矩形
    for (Rect face : faces) {
        rectangle(src, face, Scalar(255, 0, 0));
    }

    //把图片展示到Surface中
    ...

    src.release();
    gray.release();
    env->ReleaseByteArrayElements(data_, data, 0);
}

这里就是识别相关代码逻辑,NV21的结构这里不再多说,图片旋转是因为摄像头放置方向的问题,灰度化是把图片变黑白,二值化是突出轮廓,都是为了提高OpenCV识别率,相关知识也很多,有兴趣可以去深度了解OpenCV。

  1. 上面已经识别出目标,并用红色矩形框了出来,现在需要把操作完成的图片显示到Surface中。显示需要用到ANativeWindow这个类,setSurface的时候把Surface设置给window。
Java_com_yu_opencvdemo_OpencvJni_setSurface(JNIEnv *env, jobject instance, jobject surface) {

    if (window) {
        ANativeWindow_release(window);
        window = 0;
    }
    window = ANativeWindow_fromSurface(env, surface);

}

然后还是在postData的函数中,把图片中的数据(src.data)一行一行拷贝到window,还是在上面这个函数中,每一行的作用已注释:

Java_com_yu_opencvdemo_OpencvJni_postData(JNIEnv *env, jobject instance, jbyteArray data_,
                                          jint width, jint height, jint cameraId) {
    jbyte *data = env->GetByteArrayElements(data_, NULL);

     //识别相关代码
    ...

    //显示到surface
    if (window) {
        ANativeWindow_setBuffersGeometry(window, src.cols, src.rows, WINDOW_FORMAT_RGBA_8888);
        ANativeWindow_Buffer window_buffer;
        do {
            //lock失败 直接brek出去
            if (ANativeWindow_lock(window, &window_buffer, 0)) {
                ANativeWindow_release(window);
                window = 0;
                break;
            }

            uint8_t *dst_data = static_cast<uint8_t *>(window_buffer.bits);
            //stride : 一行多少个数据
            //(RGBA) * 4
            int dst_linesize = window_buffer.stride * 4;

            //一行一行拷贝,src.data是图片的RGBA数据,要拷贝到dst_data中,也就是window的缓冲区里
            for (int i = 0; i < window_buffer.height; ++i) {
                memcpy(dst_data + i * dst_linesize, src.data + i * src.cols * 4, dst_linesize);
            }
            //提交刷新
            ANativeWindow_unlockAndPost(window);
        } while (0);


    }
    ...
    env->ReleaseByteArrayElements(data_, data, 0);
}

OK,愉快的识别吧。这里先说下,OpenCV这个自带的识别模型。。识别率不是太高。。

三、训练库

这里不想多写什么(相关知识我迷糊),但是,用着很简单,并不需要你有数据算法知识,OpenCV提供了训练工具,你只要准备好正样本和负样本,比如你想做猪脸识别,准备好各式各样的猪脸作为正样本,再准备一些比如人脸狗脸什么的作为负样本,然后命令行调用OpenCV工具就好,工具在OpenCV官网下载Window版本的里面包含。
训练文档

四、项目地址

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352