上次更新简书已经是上个月的事情了,然后更新被我自己主动停掉了,因为现在的更新目的确实已经偏离了我最开始的目的,慢慢的它变成了每天的任务,随后也就出现了敷衍,而最开始的目的其实是希望自己可以每天有些新的输入然后通过输出的方式来将他们记录下来,算是记录与分享,也想通过这种方式来提升自己的书写能力。
回归正题,最近分配了一个新需求给我,使用别人提供 so 和 .h 文件。
我的第一反应就是,啊!不给源文件么?询问后对方说,这次只给 so 和头文件,一方面处于安全考虑,另一方面是不用每次源码改动了都发你一份源码,不过他的源码确实好复杂,依赖了三个开源库,要是给我了我可能也不知道如何编译...
接下来介绍的实现方式,并不是唯一的实现方式,只是我习惯用方式,如果更想查看更多方式,还需要自己网上搜索。
之前我接触 NDK 的时候是从自己写 JNI 开始的,实现了一个很简单的 Demo,通过使用别人的 c 代码,实现了将传递的内容进行 md5 加密,但此后也就没有在使用过了。
由于代码是对方提供的,所以对方并不知道 JNI 的命名规范,方法名并不是 Java_包名类名方法名,所以这个时候我想到了一个方法,我自己写一层 JNI,在自己 JNI 中去调用他 .h 中声明的内容。
那么首先要做就是写自己的类,然后在类中写需要的本地方法,代码如下:
public class TestJNI {
public native int init(String path);
}
接下来我要做就是生成该文件的头文件,我们需要在命令行中 cd 到 src/main/jave 即可,然后执行:
javah -jni 包名.类名
例:javah -jni com.jaaaelu.gzw.testjni.TestJNI
然后你需要自己写的 .c 或 .cpp 文件,因为你需要调用别人的方法所以你肯定也就需要先在你的源文件中将别人的 .h 文件 include 进来,这时候你就可以调用别人的方法了。
当然可能在生成头文件和写源文件时可能会出错,系统提示你的项目还暂时不支持 c / c++,你需要一定进行一些配置,而我习惯使用 mk 文件,下面就是文件内容:
Android.mk (我一般创建在 App 目录下)
#表示源文件在开发树中的位置
# my-dir 将返回当前目录(包含 Android.mk 文件本身的目录)的路径
LOCAL_PATH := $(call my-dir)
#为您清除 LOCAL_XXX 变量,除了 LOCAL_PATH
include $(CLEAR_VARS)
#指定一个当前模块名
LOCAL_MODULE := testJNI
#源文件位置
LOCAL_SRC_FILES := src/main/jni/com_jaaaelu_gzw_testjni_TestJNI.cpp
#会帮你把源文件生成出对应的 so (动态库)
include $(BUILD_SHARED_LIBRARY)
Application.mk (我一般创建在 App 目录下,但一般可以没有)
#生成所有架构对应的 so
APP_ABI := all
App 下的 build.gradle
#在 defaultConfig 下声明
externalNativeBuild {
ndkBuild {
arguments "NDK_APPLICATION_MK:=Application.mk"
}
}
#在 android 下声明
externalNativeBuild {
ndkBuild {
path "Android.mk"
}
}
#最好也说明一下的 so 放的位置在哪里,因为你需要用到别人的 so
sourceSets {
main {
jniLibs.srcDirs = ['libs']
}
}
这时候我们继续编写自己的源文件内容了,比如图中这样,我在自己的代码中去调用别人的方法。
你写了对应的方法后,你还需要修改 Android.mk,因为你需要依赖别人 so,这时候也就涉及到了 so 库之间的动态链接,具体如何,其实我也不懂,你只要在 Android.mk 配置好了即可,配置内容如下:
include $(CLEAR_VARS)
LOCAL_MODULE := decoder #指定一个当前模块名
LOCAL_SRC_FILES := libs/$(TARGET_ARCH_ABI)/libdecoder.so #指定文件來源,這裡會對應起來
include $(PREBUILT_SHARED_LIBRARY) #编译目标,PREBUILT_表示已经编译好的,在使用NDK编译时不会再次编译,而是直接拷贝到libs目录
#预编译.a静态库使用 PREBUILT_STATIC_LIBRARY
#这是之前的旧代码
include $(CLEAR_VARS)
LOCAL_MODULE := testJNI
LOCAL_SRC_FILES := src/main/jni/com_jaaaelu_gzw_testjni_TestJNI.cpp
LOCAL_SHARED_LIBRARIES := decoder
#可能会引发“未定义的符号”错误,加上这句即可
LOCAL_ALLOW_UNDEFINED_SYMBOLS := true
include $(BUILD_SHARED_LIBRARY)
上面的代码有两点需要注意的是 $(TARGET_ARCH_ABI) 正常情况可能我们会写 arm64-v8a 或 armeabi,如果是这样写他会去找对应的架构目录下的 so,还有一点就是可能会引发“未定义的符号”错误,加上 LOCAL_ALLOW_UNDEFINED_SYMBOLS := true 即可。
最后一步,别忘了你自己的模块需要加载:
static {
System.loadLibrary("testJNI");
}
基本上的流程就是这样,将别人的 so 和 .h 引入,然后写一层自己的 JNI,从而调用别人的方法,但是对应的一些内容你也需要配置好,如果有任何问题可以随时和我交流,也可以参考官方的文档:
https://developer.android.com/ndk/guides/prebuilts.html?hl=zh-cn