OpenCV强大的图像处理被广泛应用与各行各业,如何将其部署到不同的平台是每个OpenCV开发人员必须面对解决的。对于OpenCV开发,很多选择C++实现基本的算法,而如何将算法应用到IOS、Android或嵌入式设备上却 是一个问题,这节将讲解OpenCV 在Android上的部署,这里选择在Android Studio上开发Android应用程序。

1.首先在Opencv官网下载Android SDK,这里选中最新的opencv-3.4.1-android-sdk.zip下载 ,解压得到OpenCV-android-sdk。

2.启动Android-stiudio(本人安装的是AS 3.1版本),新建一个Android项目,Application name填OpencvJNI,Company domaint填xinyi61,Package name者为xinyi61.opencvjni,并且选上Include C++ support,最终效果如下:

OPENCV 大米 opencv jni_OPENCV 大米

3.选中app/src/main右键 New->Folder->JNI Folder,这将在main下面创建一个jni的目录,将1解压文件夹下OpenCV-android-sdk/sdk下的native目录整个拷贝到jni目录下,此时jni目录展开如下:

OPENCV 大米 opencv jni_OPENCV 大米_02

4. 选中app/src/main/java/xinyi61.opencvjni 右键New->Java Class,新建一个OpencvJNI的java类,OpencvJNI如下所示

public class OpencvJNI {
    static {
        System.loadLibrary("imgproc");
    }

    public int api_method;
    public int img_width;
    public int img_height;
    public int retval;
    public int [] srcbuf;
    public int [] dstbuf;
    public int [] outmuf;
    public int [] inmask;
    public double clarityVal;
    public String version;

    public static native Object jniApiMethod(Object param);
}

5.编写带有native声明的方法的java类,使用 javac 命令编译所编写的java类。点击Build->Make Project,完成后会在app/build/intermediates/classes/debug/xinyi61/opencvjni下生成一个OpencvJNI.class,如下所示

OPENCV 大米 opencv jni_OPENCV 大米_03

6.使用 “ javah -jni java类名”  生成扩展名为h的头文件,使用C/C++实现本地方法。打开Android Studio的Terminal,cd 到app/src/main/jni目录下,输入

javah -jni -classpath ../../../build/intermediates/classes/debug xinyi61.opencvjni.OpencvJNI

自动生成头文件,查看jni下生成的头文件xinyi61_opencvjni_OpencvJNI.h,

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class xinyi61_opencvjni_OpencvJNI */

#ifndef _Included_xinyi61_opencvjni_OpencvJNI
#define _Included_xinyi61_opencvjni_OpencvJNI
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     xinyi61_opencvjni_OpencvJNI
 * Method:    jniMethod
 * Signature: (Ljava/lang/Object;)Ljava/lang/Object;
 */
JNIEXPORT jobject JNICALL Java_xinyi61_opencvjni_OpencvJNI_jniMethod
  (JNIEnv *, jclass, jobject);

#ifdef __cplusplus
}
#endif
#endif

 

选中jni右键New->C/C++ Source FIle 命名为xinyi61_opencvjni_OpencvJNI,作为对应的java类的JNI实现源文件,其中需要访问java中定义的类对象的实例,则由以下几步完成

 

1.通过GetObjectClass()函数获得该对象的类class,返回值是一个jclass;
2.调用GetFieldID函数得到需要访问的实例域(变量)在该c类中id
3.调用Get【Type】Field()得到访问的变量值。其中【Type】对应变量的类型。

最终实现为:

//
// Created by Administrator on 2018/7/3/003.
//
#include <jni.h>
#include <stdio.h>
#include <stdlib.h>
#include <string>
#include <iostream>
#include <android/log.h>
#include "opencv2/opencv.hpp"
#include "xinyi61_opencvjni_OpencvJNI.h"

#define   LOG_TAG    "MYJNI"
#define   LOGI(...)  __android_log_print(ANDROID_LOG_INFO, LOG_TAG, __VA_ARGS__)
#define   LOGE(...)  __android_log_print(ANDROID_LOG_ERROR, LOG_TAG, __VA_ARGS__)

#ifdef __cplusplus
extern "C" {
#endif

enum JNI_API__METHOD
{
    JNI_API_METHOD_GETVERSION=0,        /*get build version and time*/
    JNI_API_METHOD_IM2GRAY,             /* convert image to gray*/
    JNI_API_METHOD_INPAINT,             /* image restore */
    JNI_API_METHOD_AUTOCOMPLATE,        /* image auto restore */
    JNI_API_METHOD_AUTOWHITEBALANCE,    /* image auto white balance */
    JNI_API_METHOD_AUTOWHITEBACKGROUND, /* replace background with white */
    JNI_API_METHOD_ADJUSTIMAGECLARITY,  /* adjust image clarity*/
    JNI_API_METHOD_INTERACTIVEMATTING,   /* interactivate matting */
    JNI_API_METHOD_END = JNI_API_METHOD_INTERACTIVEMATTING
};

JNIEXPORT jobject JNICALL Java_bc_imgproc_jniapi_jniApiMethod
        (JNIEnv *env, jclass jcls, jobject jobj)
{
   jclass jclsinput = env->GetObjectClass(jobj);

    if (jclsinput == NULL)
    {
        return env->NewStringUTF("can not find class");
    }

    /* get jobj property id */
    jfieldID jfdmethod     = env->GetFieldID(jclsinput, "api_method", "I");
    jfieldID jfdimgwidth   = env->GetFieldID(jclsinput, "img_width", "I");
    jfieldID jfdimgheight  = env->GetFieldID(jclsinput, "img_height", "I");
    jfieldID jfdcallret    = env->GetFieldID(jclsinput, "retval", "I");
    jfieldID jfdsrc        = env->GetFieldID(jclsinput, "srcbuf", "[I");
    jfieldID jfddst        = env->GetFieldID(jclsinput, "dstbuf", "[I");
    jfieldID jfdinmask     = env->GetFieldID(jclsinput, "inmask", "[I");
    jfieldID jfdversion    = env->GetFieldID(jclsinput, "version", "Ljava/lang/String;");
    jfieldID jfdclarityVal = env->GetFieldID(jclsinput, "clarityVal", "D");
    /* create new object for return */
    jobject jobjret = env->AllocObject(jclsinput);
    /* get property value,get jni method */
    int japi_call_method = env->GetIntField(jobj, jfdmethod);
    LOGI("japi_call_method:%d", japi_call_method);
    if ((japi_call_method < JNIAPI_METHOD_VERSION) || (japi_call_method > JNIAPI_METHOD_END))
    {
        LOGE("RET_ERROR_JNI_METHOD_ISNOT_SURPPORT:%x", RET_ERROR_JNI_METHOD_ISNOT_SURPPORT);
        env->SetIntField(jobjret, jfdcallret, RET_ERROR_JNI_METHOD_ISNOT_SURPPORT);
        return jobjret;
    }

    if (japi_call_method == JNIAPI_METHOD_VERSION)
    {
        char buf[40];
        sprintf(buf, "V:%s Build Time:%s-%s", CV_VERSION, __DATE__, __TIME__);
        std::string version = buf;
        jstring     jstr    = env->NewStringUTF(version.c_str());
        if (jstr == NULL)
        {
            env->SetIntField(jobjret, jfdcallret, RET_FAILED);
            return jobjret;
        }

        env->SetObjectField(jobjret, jfdversion, jstr);
        env->SetIntField(jobjret, jfdcallret, RET_OK);
        return jobjret;
    }

    int ret = 0;
    /* get property value,img_width img_height */
    int w = env->GetIntField(jobj, jfdimgwidth);
    int h = env->GetIntField(jobj, jfdimgheight);
    if ((w < MINIMUM_IMAGE_WIDTH) || (h < MINIMUM_IMAGE_HEIGHT))
    {
        env->SetIntField(jobjret, jfdcallret, RET_ERROR_IAMGE_ISNOT_SURPPORT);
        return jobjret;
    }

    LOGI("00japi_call_method:%d", japi_call_method);
    jintArray jint_bufsrc = (jintArray)env->GetObjectField(jobj, jfdsrc);
    /* get srcbuf length */
    jsize len = env->GetArrayLength(jint_bufsrc);
    if (len <= 0)
    {
        env->SetIntField(jobjret, jfdcallret, RET_ERROR_IAMGE_LEN_ERROR);
        return jobjret;
    }

    /* get point to array object */
    jint    *bufsrc = env->GetIntArrayElements(jint_bufsrc, NULL);
    cv::Mat srcimg(h, w, CV_8UC4, (unsigned char *)bufsrc);
    cv::cvtColor(srcimg, srcimg, cv::COLOR_RGBA2BGR);
	cv::Size sz = srcimg.size();
    //cv::imwrite("/storage/emulated/0/imsrc.jpg", srcimg);
    LOGI("01srcimg size:%d len:%d w:%d h:%d", srcimg.size().area(), len, w, h);

    bc_jniapi_method method = (bc_jniapi_method)japi_call_method;
    bc_jniapi_in     inparam;
    inparam.srcimg = srcimg;
    bc_jniapi_out outparam;
    switch (japi_call_method)
    {
    case JNIAPI_METHOD_WHITEBALANCE:
        break;

    case JNIAPI_METHOD_INPAINT:
       {
           jintArray jint_bufinmask = (jintArray)env->GetObjectField(jobj, jfdinmask);
           jint      *bufinmask     = env->GetIntArrayElements(jint_bufinmask, NULL);
           inparam.maskimg = cv::Mat(h, w, CV_8UC1, (unsigned char *)bufinmask);
           break;
       }
    }

    LOGI("02bc_algorithm_api call:%d", japi_call_method);
    ret = bc_algorithm_api(method, &inparam, &outparam);
    LOGI("111bc_algorithm_api call end ret:%d result mat:%zu %d", ret, outparam.dstimg.size(), outparam.dstimg[0].channels());
    /* new an jintArray object */
    jintArray retdst = env->NewIntArray(outparam.dstimg.size() * w * h);
    if (retdst == NULL)
    {
        env->SetIntField(jobjret, jfdcallret, RET_MALLOC_MEMORY_ERROR);
        return jobjret;
    }

    LOGI("03bc_algorithm_api call:%d", japi_call_method);
    jint *dbuf = env->GetIntArrayElements(retdst, NULL);
    for (int i = 0; i < outparam.dstimg.size(); ++i)
    {
        cv::cvtColor(outparam.dstimg[i], outparam.dstimg[i], cv::COLOR_BGR2RGBA);
		cv::resize(outparam.dstimg[i],outparam.dstimg[i], sz, cv::INTER_AREA);
        //char buf[60];
        //sprintf(buf, "/storage/emulated/0/imresut_%d.jpg", i);
        //LOGI("return result mat rows:%d cols:%d channels:%d",
        //     outparam.dstimg[i].rows, outparam.dstimg[i].cols, outparam.dstimg[i].channels());
        //cv::imwrite(buf, outparam.dstimg[i]);
		env->SetIntArrayRegion(retdst, i * w * h,  w * h, (int *)outparam.dstimg[i].data);
    }
	
    env->SetObjectField(jobjret, jfddst, retdst);
    env->SetIntField(jobjret, jfdcallret, ret);
    env->ReleaseIntArrayElements(jint_bufsrc, bufsrc, 0);
    env->ReleaseIntArrayElements(retdst, dbuf, 0);

    LOGI("0000000api_call_method call end with ret:%d", ret);
    return jobjret;
}


#ifdef __cplusplus
}
#endif

其中java数据类型和c数据类型关系如下表所示

OPENCV 大米 opencv jni_OPENCV 大米_04

7. 将C/C++编写的文件生成动态连接库。选中jni右键New->File 建立jni的Android.mk,

LOCAL_PATH:=$(call my-dir)

include $(CLEAR_VARS)
OpenCV_INSTALL_MODULES := on
OpenCV_CAMERA_MODULES := off
OPENCV_LIB_TYPE :=STATIC

ifeq ("$(wildcard $(OPENCV_MK_PATH))","")
include $(LOCAL_PATH)/native/jni/OpenCV.mk
else
include $(OPENCV_MK_PATH)
endif
LOCAL_MODULE := opencv_world
LOCAL_SRC_FILES := xinyi61_opencvjni_OpencvJNI.cpp
LOCAL_LDLIBS +=  -lm -llog
include $(BUILD_SHARED_LIBRARY)
选中jni右键New->File 建立jni的Application.mk,
APP_STL := gnustl_static
APP_CPPFLAGS := -frtti -fexceptions
APP_PLATFORM := android-14

直接到Terminal下的jni目录运行ndk-build,此时就会在app/src/main/libs下生成各个平台的libopencv_world.so 。
libopencv_world.so 就可以拿来使用了。
这里简单写个demo程序进行验证,最后结果图如下所示:

 

8.修改app/src/main/res/layout/activity_main.xml,修改后为下

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent">

        <TextView
        android:id="@+id/processTime"
        android:layout_width="fill_parent"
        android:layout_height="20dp"
        android:text="" />

    <LinearLayout
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:orientation="horizontal">

        <Button
            android:id="@+id/chooseImg"
            android:layout_width="187dp"
            android:layout_height="wrap_content"
            android:text="选图" />

        <Button
            android:id="@+id/processImg"
            android:layout_width="187dp"
            android:layout_height="wrap_content"
            android:text="处理" />
    </LinearLayout>

    <ImageView
        android:id="@+id/imageView"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent"
        android:layout_weight="1" />
    <TextView
        android:id="@+id/bccvVersion"
        android:layout_width="fill_parent"
        android:layout_height="20dp"
        android:text="" />
</LinearLayout>

OPENCV 大米 opencv jni_OPENCV 大米_05


9,选图按钮实现从手机中选中一副图片,这就需要给手机开权限,到app/src/main/res下修改AndroidManifest.xml,修改后的文件

如下,加粗体部分为添加的部分。

 

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="xinyi61.opencvjni">
   

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

</manifest>

 

10.编辑app/src/main/java/xinyi61.opencvjni/MainActivity下实现整个处理,

package xinyi61.opencvjni;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.widget.TextView;
import android.content.Context;
import android.content.Intent;
import android.database.Cursor;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Bitmap.Config;
import android.net.Uri;
import android.provider.MediaStore;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.TextView;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.Date;
public class MainActivity extends AppCompatActivity {

    // Used to load the 'native-lib' library on application startup.
    static {
        System.loadLibrary("native-lib");
        System.loadLibrary("opencv_world");
    }

    private static final int IMAGE = 1;
    private TextView processTime;
    private TextView bccvVersion;
    private ImageView imageView;
    private Button btnChooseImg;
    private Bitmap yourSelectedImage = null;

    private Process myjni = new Process();
    private Process res = new Process();
    private static final String TAG = "bccv";
    Bitmap bm;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        processTime = (TextView) findViewById(R.id.processTime);
        bccvVersion = (TextView) findViewById(R.id.bccvVersion);
        imageView = (ImageView) findViewById(R.id.imageView);

        btnChooseImg = (Button) findViewById(R.id.chooseImg);
        btnChooseImg.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                switch (v.getId()) {
                    case R.id.chooseImg: {
                        Intent intent = new Intent(Intent.ACTION_PICK, null);
                        intent.setDataAndType(MediaStore.Images.Media.EXTERNAL_CONTENT_URI,"image/*");
                        startActivityForResult(intent, 0x1);
                        break;
                    }
                }
            }
        });

        Button processImg = (Button) findViewById(R.id.processImg);
        processImg.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                switch (v.getId()) {
                    case R.id.processImg: {
                        imageView.setDrawingCacheEnabled(true);
                        //Bitmap bitmap = imageView.getDrawingCache();
                        int w = bm.getWidth();
                        int h = bm.getHeight();
                        int[] pix = new int[w * h];
                        bm.getPixels(pix, 0, w, 0, 0, w, h);
                        long start_time = System.currentTimeMillis();
                        /* call jni api */
                        myjni.api_method =7;
                        myjni.srcbuf = new int[w * h];
                        bm.getPixels(myjni.srcbuf, 0, w, 0, 0, w, h);
                        myjni.img_width = w;
                        myjni.img_height = h;
                        Log.e(TAG, "111111111111111111a:" + myjni.api_method + " w:" + w + " h:" + h);
                        //res = (Process)myjni.jniApiMethod(myjni);
                        res =(Process)com.example.imageproc.Process.jniApiMethod(myjni);
                        if (myjni.api_method == 0)
                        {
                            bccvVersion.setText(res.version);
                            break;
                        }
                        else
                        {
                            Bitmap bm = Bitmap.createBitmap(w, h, Bitmap.Config.ARGB_8888);
                            bm.setPixels(res.dstbuf, 0, w, 0, 0, w, h);
                            long end_time = System.currentTimeMillis();
                            long procss_time = end_time - start_time;

                            imageView.setImageBitmap(bm);
                            imageView.setDrawingCacheEnabled(false);
                            Log.e(TAG, "process time:" + procss_time);
                            processTime.setText("w:" + w + " h:" + h + " ProcessTime:" + procss_time + "ms");
                            bccvVersion.setText(res.version);
                            break;
                        }
                    }
                }
            }
        });
    }

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        /* get image path */
        if (requestCode == IMAGE && resultCode == RESULT_OK && data != null) {
            Uri selectedImage = data.getData();
            String[] filePathColumns = {MediaStore.Images.Media.DATA};
            Cursor c = getContentResolver().query(selectedImage, filePathColumns, null, null, null);
            c.moveToFirst();
            int columnIndex = c.getColumnIndex(filePathColumns[0]);
            String imagePath = c.getString(columnIndex);
            showImage(imagePath);
            c.close();
        }
    }

    /* load image */
    private void showImage(String imaePath){
        bm = BitmapFactory.decodeFile(imaePath);
        ((ImageView)findViewById(R.id.imageView)).setImageBitmap(bm);
    }
}

 

最终运行效果图如下:

 

OPENCV 大米 opencv jni_bc_06

 

备注:第一次接触Android平台,都是各种baidu完成的,很多Java上的术语不知道怎么说,只能以记流水的形式记录下整个实现过程,以备后期参考,各位看官勿喷。