opencv android jni,OpenCV - AndroidStudio的JNI工程及引用OpenCV

一把利刃,用不好,會傷到你遍體鱗傷。用得好,便為你披荊斬棘,所向披靡。好與不好之間,便是歷練。

幾經波折,終于跌跌撞撞,集成了OpenCV,并實現了灰度圖片,自此一扇新的大門已經打開。

至此我手中已經基本集齊了所需的技能碎片。本文你包括:

[1].OpenCV在AndroidStudio中的集成

[2].第一個JNI項目的解析

[3].JNI中對于Android中的Bitmap類的使用

[4].一個灰度的例子開啟OpenCV的世界

1、創建項目

1.1:下載OpenCV的SDK

0b2394a2c81f6fbf1af3f94b123d267a.png

so文件所在: sdk -> native -> libs

c++的代碼 : sdk -> native -> jni -> include -> opencv2

1.2:創建一個Android Native c++的項目

項目結構如下

3c526237e08f6540fbd28a12dd36d475.png

1.3:運行第一個項目

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

3b96d1c15d50a7cbf1b093d9243e3f22.png

2.JNI初始項目分析

2.1:MainActivity分析

在靜態代碼塊中使用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();

}

2.2:native-lib.cpp分析

引入了jni和string頭文件,一個Java_com_toly1994_rec_MainActivity_stringFromJNI函數

函數體中定義了一個sring變量,并通過env指針創建了一個字符串并返回

#include

#include

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文件生成共享庫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} )

3.集成OpenCV

3.1:庫的導入及引用

將需要的庫以及so包拷貝到項目中,以及CMakeLists.txt的配置

be2acfda7f9af6e27357b1119d3a5dfb.png

#指定 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})

3.2:幾乎斷送我ndk生涯的bug

dlopen failed: library "libc++_shared.so" not found

這個bug如噩夢般卡在我ndk前行的路上,以致我幾乎放棄,五天后,終得解法:

933dc6aa6379e8ac03f380d9b674ef95.png

---->[app/build.gradle]----

android {

defaultConfig {

externalNativeBuild {

cmake {

cppFlags ""

arguments "-DANDROID_STL=c++_shared"//使用c++_shared.so

}

}

}

3.3:創建bitmap的工具類

C++中無法直接操作Android的Bitmap類,所以需要轉化為像素矩陣處理,這里先寫成頭文件。

關于#include <android/xxxxx>的飄紅,需要 build --> Refresh Linked C++ Projects

86d344ecfc4d9e17254f7a857fb7f1f3.png

---->[cpp/bitmap_utils.h]----

#ifndef REC_UTILS_H

#define REC_UTILS_H

#include

#include

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

4.OpenCV實現灰度圖片

660dbbb54e2889e130c834a17815d04b.png

4.1:下面是三個方法的具體實現

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;

}

4.2:在MainActivity中的操作:

布局很簡單,就不貼了,一個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);

}

4.3:C++文件中的處理

將圖片的像素信息灰度化盛放在dstMat,再使用dstMat創建一個Bitmap對象,至此一個邏輯就通暢了

---->[cpp/native-lib.cpp]---

#include

#include

#include

#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對象

}

至此,本篇結束。`簡單必有簡單的成本,復雜必有復雜的價值。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/538770.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/538770.shtml
英文地址,請注明出處:http://en.pswp.cn/news/538770.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

python語句int_python中的int函數

int函數用來把其他類型向下轉換為整數類型 int(3.6) --- 3 int(3.2) --- 3 int(1/3) --- 0 int("1") --- 1 int()可以按照指定的進制的整數轉換為十進制的整數&#xff0c;如果不指定默認是十進制轉換為十進制&#xff0c;第二個參數給0和不給一樣&#xff0c;除非你…

13.JAVA基礎:八進制,十六進制表示

原文路徑&#xff1a;http://simon-c.iteye.com/blog/802149 點擊查看原文 ----------------------------------------------------------- 可以使用八進制、十六進制的類型&#xff1a;long, int, short, byte 八進制表示&#xff1a;以0作為前綴&#xff0c;數碼有0,1,…

excel圖表交互聯動_深入講解EasyShu圖表與引用數據動態聯動功能

EasyShu一開始的架構是將制作好的圖表最終返回給用戶&#xff0c;不依賴用戶工作表的單元格區域引用&#xff0c;可滿足圖表繪制后的脫離數據源分享傳播&#xff0c;無奈用戶最強烈的反饋是要求圖表與數據保持聯動&#xff0c;這一需求實在對EasyShu是一個巨大的挑戰。為了將Ea…

使用pipenv代替virtualenv管理python包

前言 第一次接觸到 pipenv 是因為看到董明偉大神的《使用pipenv管理你的項目》&#xff0c;之前可能和大家的選擇類似使用 virtualenv 或者 pyenv 來管理 python 的包環境。virtualenv 是針對python的包的多版本管理&#xff0c;通過將python包安裝到一個模塊來作為python的包虛…

android代碼生成excel,AndroidExcel

Android_Excel在android中生成excel##效果圖##初始化數據首先我們要先造下測試數據&#xff0c;這里我把數據寫死在一個常量類Const中&#xff0c;如下&#xff1a;public class Const {public interface OrderInfo{public static final String[][] orderOne new String[][] {…

14.JAVA整型變量

表示形式 Java語言整型常數的3種表示形式&#xff1a; 1、十進制整數&#xff1a;99&#xff0c;-100,0 2、八進制整數&#xff1a;要以0開頭&#xff0c;如015 3、十六進制整數&#xff1a;以0x開頭&#xff0c;如0x15 點擊查看十進制八進制十六進制概念 public static v…

exists sql用法_干貨!SQL性能優化,書寫高質量SQL語句

寫SQL語句的時候我們往往關注的是SQL的執行結果&#xff0c;但是是否真的關注了SQL的執行效率&#xff0c;是否注意了SQL的寫法規范&#xff1f;以下的干貨分享是在實際開發過程中總結的&#xff0c;希望對大家有所幫助&#xff01;1. limit分頁優化當偏移量特別大時&#xff0…

python docker自動化_「docker實戰篇」python的docker爬蟲技術-移動自動化控制工具appium工具(17)...

原創文章&#xff0c;歡迎轉載。轉載請注明&#xff1a;轉載自IT人故事會&#xff0c;謝謝&#xff01;原文鏈接地址&#xff1a;「docker實戰篇」python的docker爬蟲技術-移動自動化控制工具appium工具(17)Appium是一個開源測試自動化框架&#xff0c;可用于原生&#xff0c;混…

一些adb的常用命令

顯示正在運行的服務 adb shell dumpsys activity services [packageName] 打開一個Activity adb shell am start -n &#xff5b;包(package)名&#xff5d;/Activity絕對路徑(ex:com.xxx.xxx.xxxActivity) 以調試模式啟動一個Activity adb shell am start -D -n &#xff5b;包…

android 調用restful,android調用springmvc寫的restful

下載srpingmvc的相關jarhttp://www.cnblogs.com/liuhongfeng/p/4919963.html配置spingmvc和相關接口http://blog.csdn.net/jianyuerensheng/article/details/51258942如果報錯&#xff0c;檢查JDK版本是否和本地的一致在UserController.jave中添加接口package com.zjn.controll…

eureka server配置_springcloud項目搭建第三節:eureka集群

在上一節搭建的項目基礎上&#xff0c;在創建一個eureka-server-two的子項目和eureka-server項目一樣&#xff0c;然后修改各自項目的application.yml文件eureka-server項目的application.yml文件修改2點1.修改eureka的注冊地址改成另一個eureka-server-two項目的注冊中心地址2…

15.浮點類型

數值范圍 float類型又被稱為單精度類型&#xff0c;尾數可以精確到7位有效數字&#xff0c;在很多情況下&#xff0c;float類型的精度很難滿足需求。 double類型又被稱為雙精度類型&#xff0c;尾數可以精確到14位有效數字。 浮點類型默認是double。 public static void main(…

c4d流體插件_(圖文+視頻)C4D野教程:TFD、XP和RF三大流體插件協作案例

在逛INS的時候&#xff0c;看見國外一位叫做BastardFilms的大神制作了很多流體的效果&#xff1a;尤其是他制作的很多液態煙霧的流體&#xff0c;我特別喜歡&#xff0c;由于我不知道怎么下載INS的視頻&#xff0c;所以這里只有發個截圖大家看看&#xff1a;作者這里有說明是用…

form表單用js提前執行函數若不成功則不提交_面試必會的重復提交 8 種解決方案!...

重復提交看似是一個小兒科的問題&#xff0c;但卻存在好幾種變種用法。在面試中回答的好&#xff0c;說不定會有意想不到的收獲&#xff01;現把這 8 種解決方案分享給大家&#xff01;1.什么是冪等在我們編程中常見冪等select查詢天然冪等delete刪除也是冪等,刪除同一個多次效…

16.char類型

char&#xff0c;占2個字節。 單引號用來表示字符常量。例如a. char類型用來表示在Unicode編碼表中的字符。 unicode編碼被設計用來處理各種語言的所有文字&#xff0c;它占了2個字節&#xff0c;可允許有65536個字符。 Java語言中還允許使用轉義字符‘\&#xff0c;來將其后的…

轉向Kotlin——數據類和封閉類

數據類和封閉類是Kotlin中的兩種特殊的類&#xff0c;今天一起了解一下。更多精彩內容也可以關注我的微信公眾號——Android機動車 數據類 數據類是Kotlin的一個語法糖。Kotlin編譯器會自動為數據類生成一些成員函數&#xff0c;以提高開發效率。 數據類的使用 無論是Java服務器…

idea前端可視化_jsp可視化開發工具_netbeans jsp可視化_idea 可視化開發 jsp

數字生態鉅惠來襲&#xff01;秒殺 2核4G 5M帶寬 1200元/3年&#xff0c;1核1G首購 99元/年把默認改成 myeclipse jsp editor()原默認的jsp編輯器是 myeclipse visual jspdesigner&#xff0c;顧名思義&#xff0c;此編譯器是jsp可視化編輯器&#xff0c;對于初學者有很多的幫助…

開博爾智能android播放器,高端安卓播放器的選擇——開博爾Q10Plus 二代 4K高清播放器...

隨著OPPO和三星相繼宣布退出4K藍光播放器市場&#xff0c;先鋒的新機遲遲無法大量鋪貨&#xff0c;現在市面上可選擇的4K播放器就比較少了&#xff0c;價格也很高了&#xff0c;于是很多人開始將注意力轉向了安卓機&#xff0c;其中開博爾是比較有代表性的廠家了&#xff0c;這…

17.類型轉換

自動類型轉換 容量小的數據類型可以自動轉換為容量大的數據類型 byte b 123;//byte b2 300;//報錯&#xff0c;超過了byte最大值127//char c -3;//報錯&#xff0c;char范圍是0~65536&#xff0c;不在范圍char c2 a;//a在ascii碼里是98int i c2;long d01 123213;float f…

docker rabbitmq_一文看懂Rabbitmq,從安裝到實戰演練

Rabbitmq的初步使用隨著微服務概念發展&#xff0c;大應用逐步拆分為小應用&#xff0c;提高開發效率&#xff0c;專門的人做專門的事情&#xff0c;逐漸的流行起來。在微服務上實現通信的方式大部分是采用rpc方式&#xff0c;也有升級版本的grpc。還有另外一種實現就是使用mq來…