溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

AndroidStudio中JNI工程及引用OpenCV的示例分析

發(fā)布時(shí)間:2021-12-13 17:24:41 來(lái)源:億速云 閱讀:118 作者:小新 欄目:移動(dòng)開(kāi)發(fā)

小編給大家分享一下AndroidStudio中JNI工程及引用OpenCV的示例分析,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!

1、創(chuàng)建項(xiàng)目
1.1:下載OpenCV的SDK

首先到官網(wǎng)下載OpenCV的Android包

AndroidStudio中JNI工程及引用OpenCV的示例分析

so文件所在: sdk -> native -> libs
c++的代碼 : sdk -> native -> jni -> include -> opencv2
1.2:創(chuàng)建一個(gè)Android Native c++的項(xiàng)目

項(xiàng)目結(jié)構(gòu)如下

AndroidStudio中JNI工程及引用OpenCV的示例分析

1.3:運(yùn)行第一個(gè)項(xiàng)目

結(jié)果如下,在中間顯示了一行:"Hello from C++"

AndroidStudio中JNI工程及引用OpenCV的示例分析

2.JNI初始項(xiàng)目分析
2.1:MainActivity分析

靜態(tài)代碼塊中使用System.loadLibrary方法加載了native-lib
native方法stringFromJNI()返回一個(gè)String并設(shè)置到了TextView上

---->[src/main/java/com/toly1994/rec/MainActivity.java]----
public class MainActivity extends AppCompatActivity {
    static {
        System.loadLibrary("native-lib");
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        TextView tv = findViewById(R.id.sample_text);
        tv.setText(stringFromJNI());
    }

    /**
     * A native method that is implemented by the 'native-lib' native library,
     * which is packaged with this application.
     */
    public native String stringFromJNI();
}
2.2:native-lib.cpp分析

引入了jni和string頭文件,一個(gè)Java_com_toly1994_rec_MainActivity_stringFromJNI函數(shù)
函數(shù)體中定義了一個(gè)sring變量,并通過(guò)env指針創(chuàng)建了一個(gè)字符串并返回

#include <jni.h>
#include <string>

extern "C" JNIEXPORT jstring JNICALL
Java_com_toly1994_rec_MainActivity_stringFromJNI(
        JNIEnv *env,
        jobject /* this */) {
    std::string hello = "Hello from C++";
    return env->NewStringUTF(hello.c_str());
}
2.3:CMakeLists.txt
#指定 cmake 的最小版本
cmake_minimum_required(VERSION 3.4.1)

# 使用native-lib.cpp文件生成共享庫(kù)native-lib
add_library(native-lib SHARED native-lib.cpp )

# 在ndk中查找log庫(kù) 取別名log-lib
find_library(log-lib log )

#設(shè)置 target 需要鏈接的庫(kù)
target_link_libraries(native-lib ${log-lib} )

3.集成OpenCV
3.1:庫(kù)的導(dǎo)入及引用

將需要的庫(kù)以及so包拷貝到項(xiàng)目中,以及CMakeLists.txt的配置

AndroidStudio中JNI工程及引用OpenCV的示例分析

#指定 cmake 的最小版本
cmake_minimum_required(VERSION 3.4.1)

include_directories(include)#引入文件夾

#編譯頭文件
#定義全局 my_source_path 變量
file(GLOB my_source_path ${CMAKE_SOURCE_DIR}/*.cpp ${CMAKE_SOURCE_DIR}/*.c)
add_library(tolyCV SHARED ${my_source_path})

#添加動(dòng)態(tài)鏈接庫(kù)
add_library(lib_opencv SHARED IMPORTED)
set_target_properties(lib_opencv PROPERTIES IMPORTED_LOCATION ${CMAKE_SOURCE_DIR}/../jniLibs/${ANDROID_ABI}/libopencv_java4.so)

# 在ndk中查找log庫(kù) 取別名log-lib
find_library(log-lib log)

# 在ndk中查找jnigraphics庫(kù) 取別名jnigraphics-lib
# jnigraphics包含圖形操作的庫(kù)
find_library(jnigraphics-lib jnigraphics)

#設(shè)置 target 需要鏈接的庫(kù)
target_link_libraries(
        tolyCV
        lib_opencv
        ${jnigraphics-lib}
        ${log-lib})
3.2:幾乎斷送我ndk生涯的bug

dlopen failed: library "libc++_shared.so" not found
這個(gè)bug如噩夢(mèng)般卡在我ndk前行的路上,以致我?guī)缀醴艞?,五天后,終得解法:

AndroidStudio中JNI工程及引用OpenCV的示例分析

---->[app/build.gradle]----
android {
    defaultConfig {
        externalNativeBuild {
            cmake {
                cppFlags ""
                arguments "-DANDROID_STL=c++_shared"//使用c++_shared.so
            }
        }
    }

3.3:創(chuàng)建bitmap的工具類

C++中無(wú)法直接操作Android的Bitmap類,所以需要轉(zhuǎn)化為像素矩陣處理,這里先寫成頭文件。
關(guān)于#include &lt;android/xxxxx&gt;的飄紅,需要 build --&gt; Refresh Linked C++ Projects

AndroidStudio中JNI工程及引用OpenCV的示例分析

---->[cpp/bitmap_utils.h]----
#ifndef REC_UTILS_H
#define REC_UTILS_H

#include <android/bitmap.h>
#include <opencv2/opencv.hpp>

using namespace cv;//Mat

extern "C" {
    /**
     * Bitmap 轉(zhuǎn)矩陣
     * @param env JNI環(huán)境
     * @param bitmap Bitmap對(duì)象
     * @param mat 圖片矩陣
     * @param needPremultiplyAlpha 是否前乘透明度
     */
void bitmap2Mat(JNIEnv *env, jobject bitmap, Mat *mat, bool needPremultiplyAlpha = false);

/**
 * 矩陣轉(zhuǎn)Bitmap
 * @param env JNI環(huán)境
 * @param mat 圖片矩陣
 * @param bitmap Bitmap對(duì)象
 * @param needPremultiplyAlpha 是否前乘透明度
 */
void mat2Bitmap(JNIEnv *env, Mat mat, jobject bitmap, bool needPremultiplyAlpha = false);

/**
 *
 * 創(chuàng)建Bitmap
 * @param env JNI環(huán)境
 * @param src 矩陣
 * @param config Bitmap配置
 * @return Bitmap對(duì)象
 */
jobject createBitmap(JNIEnv *env, Mat src, jobject config);
}

#endif //REC_UTILS_H
4.OpenCV實(shí)現(xiàn)灰度圖片

AndroidStudio中JNI工程及引用OpenCV的示例分析

4.1:下面是三個(gè)方法的具體實(shí)現(xiàn)

bitmap2Mat 通過(guò)bitmap獲取像素矩陣,放入mat中,這樣mat就可以在C++中操作
mat2Bitmap 與上面相反,通過(guò)將mat矩陣,將矩陣的像素信息置入其中
createBitmap 通過(guò)反射獲取Android中的createBitmap方法獲取對(duì)象,在通過(guò)mat2Bitmap置入信息。

#include "bitmap_utils.h"

void bitmap2Mat(JNIEnv *env, jobject bitmap, Mat *mat, bool needPremultiplyAlpha) {
    AndroidBitmapInfo info;
    void *pixels = 0;
    Mat &dst = *mat;
    //獲取信息和一些斷言
    CV_Assert(AndroidBitmap_getInfo(env, bitmap, &info) >= 0);//獲取Bitmap信息
    CV_Assert(info.format == ANDROID_BITMAP_FORMAT_RGBA_8888//圖片格式RGBA_8888 或RGB_565
              || info.format == ANDROID_BITMAP_FORMAT_RGB_565);
    CV_Assert(AndroidBitmap_lockPixels(env, bitmap, &pixels) >= 0);
    CV_Assert(pixels);

    dst.create(info.height, info.width, CV_8UC4);
    if (info.format == ANDROID_BITMAP_FORMAT_RGBA_8888) {
        Mat tmp(info.height, info.width, CV_8UC4, pixels);
        if (needPremultiplyAlpha) {
            cvtColor(tmp, dst, COLOR_mRGBA2RGBA);
        } else {
            tmp.copyTo(dst);
        }
    } else {
        Mat tmp(info.height, info.width, CV_8UC2, pixels);
        cvtColor(tmp, dst, COLOR_BGR5652RGBA);
    }
    AndroidBitmap_unlockPixels(env, bitmap);
}

void mat2Bitmap(JNIEnv *env,  Mat mat, jobject bitmap,bool needPremultiplyAlpha) {
    AndroidBitmapInfo info;
    void *pixels = 0;
    CV_Assert(AndroidBitmap_getInfo(env, bitmap, &info) >= 0);//獲取Bitmap信息
    CV_Assert(info.format == ANDROID_BITMAP_FORMAT_RGBA_8888//圖片格式RGBA_8888 或RGB_565
              || info.format == ANDROID_BITMAP_FORMAT_RGB_565);
    CV_Assert(mat.dims==2&&info.height==(uint32_t)mat.rows && info.width==(uint32_t)mat.cols);
    CV_Assert(mat.type()==CV_8UC1||mat.type()==CV_8UC3||mat.type()==CV_8UC4);
    CV_Assert(AndroidBitmap_lockPixels(env, bitmap, &pixels) >= 0);
    CV_Assert(pixels);

    if (info.format == ANDROID_BITMAP_FORMAT_RGBA_8888) {
        Mat tmp(info.height, info.width, CV_8UC4, pixels);
        switch (mat.type()){
            case CV_8UC1:
                cvtColor(mat,tmp,COLOR_GRAY2RGBA);
                break;
            case CV_8UC3:
                cvtColor(mat,tmp,COLOR_RGB2RGBA);
                break;
            case CV_8UC4:
                cvtColor(mat,tmp,COLOR_RGBA2mRGBA);
                if (needPremultiplyAlpha) {
                    cvtColor(mat, tmp, COLOR_RGBA2mRGBA);
                } else {
                    mat.copyTo(tmp);
                }
                break;
            default:break;
        }
    } else {
        Mat tmp(info.height, info.width, CV_8UC2, pixels);
        switch (mat.type()){
            case CV_8UC1:
                cvtColor(mat,tmp,COLOR_GRAY2BGR565);
                break;
            case CV_8UC3:
                cvtColor(mat,tmp,COLOR_RGB2BGR565);
                break;
            case CV_8UC4:
                cvtColor(mat,tmp,COLOR_RGBA2BGR565);
                break;
            default:break;
        }
    }
    AndroidBitmap_unlockPixels(env, bitmap);
}

jobject createBitmap(JNIEnv *env, Mat src, jobject config){
    jclass java_bitmap_class=(jclass)env->FindClass("android/graphics/Bitmap");//類名
    jmethodID mid=env->GetStaticMethodID(java_bitmap_class,"createBitmap",//獲取方法
            "(IILandroid/graphics/Bitmap$Config;)Landroid/graphics/Bitmap;");
    jobject bitmap=env->CallStaticObjectMethod(java_bitmap_class,mid,src.cols,src.rows,config);
    mat2Bitmap(env,src,bitmap, false);
    return  bitmap;
}
4.2:在MainActivity中的操作:

布局很簡(jiǎn)單,就不貼了,一個(gè)iv_photo的ImageView。
在點(diǎn)擊時(shí)調(diào)用一個(gè)opBitmap的native方法使得圖片灰度化。

public class MainActivity extends AppCompatActivity {
    static {
        System.loadLibrary("tolyCV");
    }
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        ImageView iv = findViewById(R.id.iv_photo);
        iv.setOnClickListener(v -> {
            tv.setText(stringFromJNI());
            Bitmap bitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.wy_300x200);
            iv.setImageBitmap(opBitmap(bitmap,Bitmap.Config.ARGB_8888));
        });
    }
    public native Bitmap opBitmap(Bitmap bitmap, Bitmap.Config argb8888);
}
4.3:C++文件中的處理

將圖片的像素信息灰度化盛放在dstMat,再使用dstMat創(chuàng)建一個(gè)Bitmap對(duì)象,至此一個(gè)邏輯就通暢了

---->[cpp/native-lib.cpp]---
#include <jni.h>
#include <string>
#include <opencv2/imgproc/types_c.h>
#include "bitmap_utils.h"

extern "C"
JNIEXPORT jobject JNICALL
Java_com_toly1994_rec_MainActivity_opBitmap(JNIEnv *env, jobject instance, jobject bitmap,
                                              jobject argb8888) {
    Mat srcMat;
    Mat dstMat;
    bitmap2Mat(env, bitmap, &srcMat);
    cvtColor(srcMat, dstMat, CV_BGR2GRAY);//將圖片的像素信息灰度化盛放在dstMat
    return createBitmap(env,dstMat,argb8888);//使用dstMat創(chuàng)建一個(gè)Bitmap對(duì)象
}

以上是“AndroidStudio中JNI工程及引用OpenCV的示例分析”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對(duì)大家有所幫助,如果還想學(xué)習(xí)更多知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI