零基础带你吃掉JNI全家桶(二)

##前言 上篇通过一个简单的例子大概阐述了jni开发的基本流程,最后也编译出了自己的so文件,本篇主要介绍怎么引入第三方的so文件并进行调用

零基础带你吃掉JNI全家桶(一)

零基础带你吃掉JNI全家桶(三)

一、如何调用第三方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位的

image.png

然后放到我们新建的一个项目中,这里把so名字重命名下,防止混淆

image.png

然后我们需要在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层的类?等等。

有新的想法和疑问的老哥可以留言一起讨论哦,溜了溜了~

猜你喜欢

转载自juejin.im/post/5c6921eb6fb9a049b34885a0