您好,登錄后才能下訂單哦!
小編給大家分享一下AndroidStudio中JNI工程及引用OpenCV的示例分析,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!
首先到官網下載OpenCV的Android包
so文件所在: sdk -> native -> libs c++的代碼 : sdk -> native -> jni -> include -> opencv2
Native c++
的項目項目結構如下
結果如下,在中間顯示了一行:"Hello from C++"
在
靜態代碼塊
中使用System.loadLibrary
方法加載了native-lib
native方法stringFromJNI()
返回一個String并設置到了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(); }
native-lib.cpp
分析引入了
jni和string
頭文件,一個Java_com_toly1994_rec_MainActivity_stringFromJNI
函數
函數體中定義了一個sring變量,并通過env指針創建了一個字符串并返回
#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()); }
#指定 cmake 的最小版本 cmake_minimum_required(VERSION 3.4.1) # 使用native-lib.cpp文件生成共享庫native-lib add_library(native-lib SHARED native-lib.cpp ) # 在ndk中查找log庫 取別名log-lib find_library(log-lib log ) #設置 target 需要鏈接的庫 target_link_libraries(native-lib ${log-lib} )
將需要的庫以及so包拷貝到項目中,以及CMakeLists.txt的配置
#指定 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}) #添加動態鏈接庫 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庫 取別名log-lib find_library(log-lib log) # 在ndk中查找jnigraphics庫 取別名jnigraphics-lib # jnigraphics包含圖形操作的庫 find_library(jnigraphics-lib jnigraphics) #設置 target 需要鏈接的庫 target_link_libraries( tolyCV lib_opencv ${jnigraphics-lib} ${log-lib})
dlopen failed: library "libc++_shared.so" not found
這個bug如噩夢般卡在我ndk前行的路上,以致我幾乎放棄,五天后,終得解法:
---->[app/build.gradle]---- android { defaultConfig { externalNativeBuild { cmake { cppFlags "" arguments "-DANDROID_STL=c++_shared"//使用c++_shared.so } } }
C++中無法直接操作Android的Bitmap類,所以需要轉化為像素矩陣處理,這里先寫成頭文件。
關于#include <android/xxxxx>
的飄紅,需要build --> Refresh Linked C++ Projects
---->[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 轉矩陣 * @param env JNI環境 * @param bitmap Bitmap對象 * @param mat 圖片矩陣 * @param needPremultiplyAlpha 是否前乘透明度 */ void bitmap2Mat(JNIEnv *env, jobject bitmap, Mat *mat, bool needPremultiplyAlpha = false); /** * 矩陣轉Bitmap * @param env JNI環境 * @param mat 圖片矩陣 * @param bitmap Bitmap對象 * @param needPremultiplyAlpha 是否前乘透明度 */ void mat2Bitmap(JNIEnv *env, Mat mat, jobject bitmap, bool needPremultiplyAlpha = false); /** * * 創建Bitmap * @param env JNI環境 * @param src 矩陣 * @param config Bitmap配置 * @return Bitmap對象 */ jobject createBitmap(JNIEnv *env, Mat src, jobject config); } #endif //REC_UTILS_H
bitmap2Mat 通過bitmap獲取像素矩陣,放入mat中,這樣mat就可以在C++中操作
mat2Bitmap 與上面相反,通過將mat矩陣,將矩陣的像素信息置入其中
createBitmap 通過反射獲取Android中的createBitmap方法獲取對象,在通過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; }
布局很簡單,就不貼了,一個iv_photo的ImageView。
在點擊時調用一個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); }
將圖片的像素信息灰度化盛放在dstMat,再使用dstMat創建一個Bitmap對象,至此一個邏輯就通暢了
---->[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創建一個Bitmap對象 }
以上是“AndroidStudio中JNI工程及引用OpenCV的示例分析”這篇文章的所有內容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內容對大家有所幫助,如果還想學習更多知識,歡迎關注億速云行業資訊頻道!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。