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.mzph.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来…