正在准备中的项目里,有一部分打算直接移殖Linux开发组在之前就完成的功能,他们是使用C语言开发。考虑到维护的问题,准备让他们将代码打包成so文件,再引用到我的项目中。这样也就相当于我去引用一个第三方库,并且这个库中的代码格式也不一定是我们JNI开发时规定的命名,因此,需要通过我自己的C文件再去调用so库中的方法。

1 生成SO库

1.1 Native方法

新建项目JNISODemo,在MainActivity中定义Native方法:

public native String getString();

1.2 头文件生成

.h文件的生成是在命令行cd到main目录下,再使用javah生成。

这次是想介绍下快捷方式。

File -> Settings -> Tools -> External tools -> +

Program:   $JDKPath$\bin\javah.exe
Arguments:  -classpath . -jni -o $ModuleFileDir$\src\main\jni\$Prompt$  $FileClass$
Working directory: $ModuleFileDir$\src\main\java

在我们声明native方法的类上点击右键,javah,输入命名(我命名为Test.h),之后就会先自动创建一个jni文件夹,然后生成一个Test.h文件,copy Test.h,并将命名改为Test.c。

1.3 完成C代码

在Test.c中简单完成下我们定义的方法,返回一个字符串:

#include <jni.h>
/* Header for class com_david_jnisodemo_MainActivity */#ifndef _Included_com_david_jnisodemo_MainActivity
#define _Included_com_david_jnisodemo_MainActivity
#ifdef __cplusplus
extern "C" {
#endif
/** Class:     com_david_jnisodemo_MainActivity* Method:    getString* Signature: ()Ljava/lang/String;*/
JNIEXPORT jstring JNICALL Java_com_david_jnisodemo_MainActivity_getString(JNIEnv *env, jobject instance) {return (*env)->NewStringUTF(env, "This is a test!");
}#ifdef __cplusplus
}
#endif
#endif

1.4 CMakeList.txt

cmake_minimum_required(VERSION 3.4.1)add_library( # Sets the name of the library.Test# Sets the library as a shared library.SHARED# Provides a relative path to your source file(s).src/main/jni/Test.c )find_library(log-lib log )target_link_libraries( # Specifies the target library.Test${log-lib} )

1.5 build.gradle

ndk {ldLibs "log"//实现__android_log_printabiFilters  "armeabi-v7a" //平台配置,因为在Android上,就只写了一个
}

1.6 编译生成so

编译完成后就如下图,产生一个libTest.so的文件,这就是我们要的。把它当做Linux最后打包成的so文件。

2 导入第三方so

新建一个项目JNIUseSoDemo,项目结构如下。同样是在MainActivity中定义Native方法,生成UseSo.c。将我们上一步生成的so文件拷贝到jniLibs下(armeabi-v7a是平台)。以及上一步中的头文件也拷贝到jni下。

2.1 完成C代码

我在UseSo中getString方法去调用了so库中的getString方法。

#include <jni.h>
#include "Test.h" //so库的头文件,必须要引用!/* Header for class com_david_jniusesodemo_MainActivity */#ifndef _Included_com_david_jniusesodemo_MainActivity
#define _Included_com_david_jniusesodemo_MainActivity
#ifdef __cplusplus
extern "C" {
#endif
/** Class:     com_david_jniusesodemo_MainActivity* Method:    getString* Signature: ()Ljava/lang/String;*/
JNIEXPORT jstring JNICALL Java_com_david_jniusesodemo_MainActivity_getString(JNIEnv *env, jobject instance) {return Java_com_david_jnisodemo_MainActivity_getString(env, instance);
}#ifdef __cplusplus
}
#endif
#endif

当然还没有完成。

2.2 CMakeList.txt

在CMake中将LibTest.so导入工程

cmake_minimum_required(VERSION 3.4.1)add_library( # Sets the name of the library.UseSo# Sets the library as a shared library.SHARED# Provides a relative path to your source file(s).src/main/jni/UseSo.c )#导入第三方so包,并声明为 IMPORTED 属性,指明只是想把 so 导入到项目中
add_library( TestSHAREDIMPORTED )#指明 so 库的路径,CMAKE_SOURCE_DIR 表示 CMakeLists.txt 的路径
set_target_properties(TestPROPERTIES IMPORTED_LOCATION${CMAKE_SOURCE_DIR}/src/main/jniLibs/armeabi-v7a/libTest.so )#指明头文件路径,不然会提示找不到 so 的方法
include_directories(src/main/jni/)find_library(log-liblog )target_link_libraries( # Specifies the target library.UseSoTest${log-lib} )

2.3 build.gradle

ndk {abiFilters 'armeabi-v7a'
}

2.4 最终调用

public class MainActivity extends Activity {static {System.loadLibrary("UseSo"); //加载SO库}@Overrideprotected void onCreate(@Nullable Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);Log.e("TEST",getString());}public native String getString(); //它会调用Java_com_david_jniusesodemo_MainActivity_getString方法,然后该方法又回去调用so库中的Java_com_david_jnisodemo_MainActivity_getString方法,得到返回字符串。}

验证没有问题,导入第三方so库完成。

加载so库代码:

static {
        loadNativeLibrary();
    }
 
    private static synchronized void loadNativeLibrary() {
        System.loadLibrary("encrypt");
    }
调用的so内的方法前需要添加native关键字:

private static native byte[] e(byte[] input);
需要注意的点:一、包名要和so库中的包名一样;二、类名也要一致。
我们看看so库的头文件中的函数:

JNIEXPORT jint JNICALL Java_cn_scnu_MainActivity_num
  (JNIEnv *, jobject);
函数名为Java_包名类名函数名。所以我们创建的Android工程的包名也要一致,即cn_scnu,同时,负责加载并且提供native方法的类的类名也要相同,即MainActivity,否则调用不成功,提示找不到方法。

过程中遇到两个错误:

1.is 32-bit instead of 64-bit动态库错误

2.UnsatisfiedLinkError

有价值的解决方式:

https://segmentfault.com/a/1190000006210166

https://blog.csdn.net/xiaxiayige/article/details/68925669
————————————————

java.lang.UnsatisfiedLinkError: No implementation found for

错误的可能原因:

1、.so 文件路径问题

(1) 要么你就在main/下新建一个JniLibs文件夹,再把SDK里带的so文件目录诸如armeabi等等文件夹丢进去就可以了。

(2) 要不你就把放so文件的目录诸如armeabi直接放在放jar文件的目录libs里头,然后修改build.gradle文件的内容,添加上图或者如下

android { sourceSets{ main{ jniLibs.srcDirs=['Libs'] }
}

2、jni 调用类的路径一定要一致。也就是说 .so中函数声明涉及到的package name和class name与调用它的package name和class name不符。因此我们要改变我们工程中的package name和class name。使其与.so文件中函数签名提示的一致,在这个类中加入native方法的声明。

如: JNI接口 Java_com_netease_xtc_cloudmusic_utils_NeteaseMusicUtils_nativeInit 中,com.netease.xtc.cloudmusic.utils 代表的是 package name ,NeteaseMusicUtils 则是 class name。

而第三方提供so库的工作人员的c文件的定义JNI接口为Java_com_netease_cloudmusic_utils_NeteaseMusicUtils_nativeInit,即: package name 必须为: com.netease.cloudmusic.utils ,而class name 必须为 NeteaseMusicUtils 。

3、未被成功调用

public class Native
{static{System.loadLibrary("xxxx");}
}改为:public class Native
{Native{System.loadLibrary("xxxx");}
}

版权声明:本文为CSDN博主「buder得儿得儿以得儿以得儿得儿」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/cpcpcp123/article/details/87697569

Android JNI学习-调用第三方SO库相关推荐

  1. Android JNI学习(五)——Java与Native之间如何实现相互调用

    本章将讲述Java与Native之间如何实现相互调用.我将围绕围绕如下三点来讲解. #mermaid-svg-qeVnGlVrLWrB5ryX .label{font-family:'trebuche ...

  2. Android如何调用第三方SO库

    问题描述:Android如何调用第三方SO库: 已知条件: SO库为Android版本连接库(*.so文件),并提供了详细的接口说明: 已了解解决方案: 1.将SO文件直接放到libs/armeabi ...

  3. Android JNI学习(六)——Java与Native实战演习

    前言: 前几篇我主要介绍了jni先关的基础知识和常用API,相信看过的童靴对JNI已经有了一定的了解,如果不了解也没关系,下面我给出了链接,可以点进去学习.接下来我将实战一个完整案例,案例很简单,就是 ...

  4. Android NDK开发: 通过C/C++调用第三方so库

    文章目录 一.编写so库代码 二.安装Android NDK 三.编译so库 3.1 编辑Android.mk 3.2 编辑Application.mk 3.3 编译 四.集成到Android工程中 ...

  5. android开发打开第三方库,Android开发NDK调用三方so库

    概要 在日常开发中,android NDK的作用无外乎有两种:一种是通过调用底层C/C++的算法,提高app的运行效率:另一种则是通过C/C++的特性,或者和驱动交互等,实现一些功能性的需求.接下来将 ...

  6. android jni java调用c_Android与JNI(一) ---- Java调用C 静态调用

    第一.通过eclipse新建一个工程名为HelloJni的android工程,并编译. 第二.右键工程-->Android Tools --> Add Native Support,出现如 ...

  7. android jni java调用c,Android与JNI(一) ---- Java调用C 静态调用

    第一.通过eclipse新建一个工程名为HelloJni的android工程,并编译. 第二.右键工程-->Android Tools --> Add Native Support,出现如 ...

  8. Android JNI学习(四)——JNI的常用方法的API

    前三篇主要讲解了jni基础相关的理论知识,今天主要讲解一下JNI的常用方法的API,掌握了基本的理论知识和常用的API接下来才能更好的实战. jni的常用API大纲 再看API前,我建议大家主要结合官 ...

  9. Android开发学习之路-机器学习库(图像识别)、百度翻译

    对于机器学习也不是了解的很深入,今天无意中在GitHub看到一个star的比较多的库,就用着试一试,效果也还行.比是可能比不上TensorFlow的,但是在Android上用起来比较简单,毕竟Tens ...

  10. Android好用的第三方开源库

    记录一些对工作学习有帮助的第三方开源库 快捷入口 音频类 AudioPlay Banner类 banner 流式布局 FlowLayout 网络请求框架 RxEasyHttp okhttp-RxHtt ...

最新文章

  1. Android——实现欢迎界面的自动跳转(转)
  2. Gradle实战:发布aar包到maven仓库
  3. sql 子查询(mysql)
  4. spring mvc学习(24):配置maven环境和创建maven项目(建议收藏,超全超详细)
  5. 2014年驾考科目三考试扣分标准(细则)
  6. php数值操作,php数值计算num类简单操作示例
  7. 基于 Kubernetes 实践弹性的 CI/CD 系统
  8. 别再瞎搞了,处理Java异常的10个最佳实践
  9. Oracle 11g安装(window)的7个服务
  10. 福泉服务器维修,欢迎访问##福泉小松挖掘机维修服务##实业集团
  11. C# 关于yield return的研究(转载)
  12. word中编辑公式及公式编号的对齐设置
  13. 技巧 | Markdown 语法中首行缩进的方法
  14. android名字的由来构词,词根词缀词典_根据词缀查单词的词典_词根词缀 书_词根词缀记忆字典4.4.0 for android...
  15. 鼠标右键菜单及图标添加
  16. ArcGIS数据生产与精细化制图之中国年降水量分布图的制作
  17. Python 爬虫--下载音乐
  18. neu计算机硕士,BC省可直接移民的硕士专业-NEU-Align计算机硕士
  19. 怎样转载csdn文章
  20. vivo手机打开手机测试页面,查看电池损耗代码

热门文章

  1. Django框架学习——4—(DTL模板标签、模版常用过滤器、模版结构优化、加载静态文件)
  2. Linux和Windows双系统gpt,Windows+Ubuntu双系统GPTMBR引导设置
  3. logistic regression(二项 logistic 与 多项logistic )
  4. 牛客网C++开发面试经验汇总
  5. 【华为OJ】【MML命令执行结果查询】
  6. 香橙派借助语音模块实现语音刷抖音
  7. c语言eval函数,百行代码轻便实现C#中的Eval函数
  8. Android - scheme 一个app跳转另一个app、模块开发
  9. 三元组事件抽取与简单代码实现
  10. MYSQL建表时PK,NN,UQ,BIN,UN,ZF,AI字段标识的意义