前言
上篇通过一个简单的例子大概阐述了jni开发的基本流程,最后也编译出了自己的so文件,本篇主要介绍怎么引入第三方的so文件并进行调用
一、如何调用第三方so中的方法?
通过上篇我们知道,从Java层要调用native层的方法,要么是静态注册,要么是动态注册,但是不管是哪一种,两个方法之间必须需要建立一定的通道关系,静态注册需要对应好方法名,动态注册需要对两个方法进行绑定,都是需要知道包名的,但是问题来了,那么如何在我的项目中调用第三方so中的方法呢?自己的项目包名都是不一样的,实际上有两种方式:
- 如果第三方提供了so文件,同时也提供了SDK jar包文件,那实际上自己本身就不需要做太多的操作,直接调用API中的方法,sdk内部再去跟native方法进行映射,我们只要将so库文件导入进来放在指定位置,一般是在jniLibs目录下,这样sdk里面就可以跟native层通信了。
- 上面那种方式一般适用于集成第三方服务,比如高德、友盟等等,局限性较大,只能在sdk限制下进行操作,假如是需要对其进行一定的扩展性或者没有人给你提供SDK包(比如自己公司的一些内部库经常会有这种情况),那么这个时候就需要自己通过需要把so导进来之后,编写本地方法来进行映射通信
二、关联第三方so库
上面第一种方法就不说了,一般直接调用sdk即可,我们直接看第二种,大概分为以下几个步骤:
- 导入第三方so文件,放在指定目录下,一般就放在jniLibs目录下面
- 编写CMakeLists.txt文件,引入so库并进行关联
- 编写native方法,包名一定要和so中对应的包名一样
我们就使用上篇例子生成出来的so,我们来调用下,从build中把so拷出来,位置如下,这里我们用arm架构就行了,一个是64位的,一个是32位的
然后放到我们新建的一个项目中,这里把so名字重命名下,防止混淆
然后我们需要在build.gradle中指定so文件目录
sourceSets {
main {
jni.srcDirs = []
jniLibs.srcDirs = ['src\\main\\jniLibs']
}
}
externalNativeBuild {
cmake {
path "CMakeLists.txt"
}
}
然后,因为本身自己是不需要生成so库的,所以CMakeLists.txt中的native-lib可以删掉,我们加入要引入的so库,并与之关联
#定义cmake支持的最小版本号
cmake_minimum_required(VERSION 3.4.1)
#加入lib2库 ,定义为导入形式
add_library(lib2 SHARED IMPORTED)
#关联lib2库为我们要导入进来的libdata.so文件,ANDROID_ABI会根据自身cpu架构选择so文件
set_target_properties( lib2
PROPERTIES IMPORTED_LOCATION
${CMAKE_SOURCE_DIR}/src/main/jniLibs/${ANDROID_ABI}/libdata.so)
# 从系统里查找依赖库,可添加多个
find_library( # 例如查找系统中的log库liblog.so
log-lib
# liblog.so库指定的名称即为log,如同上面指定生成的libnative-lib.so库名称为native-lib一样
log )
这样CMakeLists文件在执行的时候就会把libdata.so文件加载进来。
最后一步,编写本地方法,新建一个类
package com.example.taolin.jni_project;
public class NativeHelper {
static {
System.loadLibrary("data");
}
public static native String stringFromJNI();
public static native int add(int a,int b);
}
主要包名,要和so中的一直的,也是上篇文章中的动态注册代码,这里粘贴部分
//动态注册
jint registerMethod(JNIEnv *env) {
jclass clz = env->FindClass("com/example/taolin/jni_project/NativeHelper");
if (clz == NULL) {
LOGD("con't find class: com/example/taolin/jni_project/NativeHelper");
}
JNINativeMethod jniNativeMethod[] = {{"stringFromJNI", "()Ljava/lang/String;", (void *) backStringToJava},
{"add", "(II)I", (void *) addNum},};
return env->RegisterNatives(clz, jniNativeMethod,
sizeof(jniNativeMethod) / sizeof(jniNativeMethod[0]));
}
这样的话,就在调用的时候,就可以找到对应的类,将Java层方法和Native层方法进行关联起来,进行通信
public class MainActivity extends AppCompatActivity {
// Used to load the 'native-lib' library on application startup.
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// Example of a call to a native method
TextView tv = (TextView) findViewById(R.id.sample_text);
tv.setText(NativeHelper.stringFromJNI());
}
/**
* A native method that is implemented by the 'native-lib' native library,
* which is packaged with this application.
*/
}
具体图我就不截了,也是能成功显示出字符串的,这样就调用了外部so的方法,大功告成!
这里只是最简单的,从native层返回一个字符串,比较浅显局限,下篇将接着介绍下,Jni的语法,以及java层和native间更为复杂的交互过程,对象怎么传输?,native层怎么操作java层的类?等等。
有新的想法和疑问的老哥可以留言一起讨论哦,溜了溜了~