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,一经查实,立即删除!

相关文章

excel图表交互联动_深入讲解EasyShu图表与引用数据动态联动功能

EasyShu一开始的架构是将制作好的图表最终返回给用户&#xff0c;不依赖用户工作表的单元格区域引用&#xff0c;可满足图表绘制后的脱离数据源分享传播&#xff0c;无奈用户最强烈的反馈是要求图表与数据保持联动&#xff0c;这一需求实在对EasyShu是一个巨大的挑战。为了将Ea…

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…

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删除也是幂等,删除同一个多次效…

开博尔智能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来…

Angular v6 正式发布

Angular 6 正式发布 Angular 6 已经正式发布了&#xff01;这个主要版本并不关注于底层的框架&#xff0c;更多地关注于工具链&#xff0c;以及使 Angular 在未来更容易快速推进。 作为发布的一部分&#xff0c;我们同步了主要的框架包 (angular/core, angular/common, angula…

py 字典添加多个value_# Python 3 # Python 3字典Dictionary(1)

Python3 字典字典是另一种可变容器模型&#xff0c;且可存储任意类型对象。字典的每个键值(key>value)对用冒号(:)分割&#xff0c;每个对之间用逗号(,)分割&#xff0c;整个字典包括在花括号({})中 ,格式如下所示&#xff1a;d {key1 : value1, key2 : value2 }键必须是唯…

饶军:Apache Kafka的过去,现在,和未来

欢迎大家前往腾讯云社区&#xff0c;获取更多腾讯海量技术实践干货哦~ 本文首发在云社区&#xff0c;未经许可&#xff0c;不得转载。大家好&#xff0c;我大概简单的介绍一下&#xff0c;我叫饶军&#xff0c;我是硅谷的初创公司Confluent的联合创始人之一&#xff0c;我们公司…

Android手机avi转换为mp4手机,如何把avi转换为mp4

AVI是目前比较常见的一种音视频交错格式&#xff0c;这种格式的文件在多种地方是可以看到的&#xff0c;大多在游戏录制、光盘文件中可以见到此类文件。其画质不错&#xff0c;适合在电脑上观看&#xff0c;但是现在我们使用较多的是手机、iPad等设备。如果我们想要在这些设备上…

伪代码书写规范_C++代码书写规范(推荐新手程序员)

代码就是程序员的面子&#xff0c;无论是在工作中在电脑上写程序代码还是在面试时在纸上写演示代码我们都希望写出整洁&#xff0c;优雅的代码。特别在工作中当我们碰到需要维护别人的代码&#xff0c;或者是多人参与一个项目大家一起写代码的时候&#xff0c;如果碰到一些丑陋…

Java基础17:Java IO流总结

版权声明&#xff1a;本文为博主原创文章&#xff0c;未经博主允许不得转载。 https://blog.csdn.net/a724888/article/details/80201802 这位大侠&#xff0c;这是我的公众号&#xff1a;程序员江湖。 分享程序员面试与技术的那些事。 干货满满&#xff0c;关注就送。 本文介…

html语言闪烁特效代码,css3 文字闪烁特效代码

今天给大家分享几个文字闪烁特效代码&#xff0c;纯css3代码实现&#xff0c;对于新手小伙伴值得拿来学习一下。文字闪烁特效一通过改变透明度来实现文字的渐变闪烁&#xff0c;代码如下&#xff1a;文字闪烁&#xff1a;闪烁效果.main{color: #666;margin-top: 50px;}/* 定义k…

2021年山西副高考试成绩查询,中国卫生人才网2021年山西卫生资格考试成绩查询...

2021年山西卫生资格考试成绩查询 由中国卫生人才网考试快讯提供&#xff0c;以及提供2021国家卫生资格成绩查询考试信息。更多关于2021年山西卫生资格考试成绩查询 ,卫生资格考试,2021卫生资格考试,卫生资格成绩快讯的内容&#xff0c;请关注国家卫生资格考试网&#xff01;!20…