JNI开发——引入第三方库并实现文件的拆分
1、引入第三方库:
(下面说的不是引入像百度地图的so库那样,百度地图除了so库还有jar包,这里是没有jar包的,要用JNI 调用第三方so库里的方法)
指定平台:在app的build.gradle中的android/defaultConfig中添加
高版本的NDK(16以上)不能包含armeabi,会报CMake Error:CMAKE_C_COMPILER not set的错误。
这是关于高版本不支持armeabi的说明指定AAR包到lib下:
指定库在项目的app/libs中
如果在src/main下创建一个jniLibs文件,把so库放在里面就不需要像上面那样指定。
2、cmake.txt 配置
这是添加一个,如果添加多个 重复上面的两个方法
如果是多个c/cpp文件把上面的src/main/cpp/native-jni.c 在下面复制一份,native-jni.c改为另外一个c/cpp的文件名。
系统在编译的时候需要链接库,所以把第三方库和我们自己写JNI生成的库都放进去,第三方库的引入就完成了(多个so库就添加多个库到方法里)
如果要本地不同的C文件生成多个so库target_link_libraries也需要多个
如果想在项目中添加系统库本来就有的库可以用这个。
3、拆分文件:
首先动态注册的方式创建了一个native_diff对应java中的diff方法。
java代码:
传入原文件的路径和子文件的路径;
c代码:
拆分之后SD卡中就会多出video_0.mp4,video_1.mp4等4个文件。
为什么申请一个二级指针:第一级存储文件名指针的地址,二级存文件名的字符。
假如是一级指针,只能表示一个文件名或每个字符不能写一个字母或数字不能写文件名,所以不能组成多个文件的数组。
4、文件的合并:
java代码 :只是增加了一个方法
C代码:
5、java线程和jni线程:
java方法和jni方法运行在同一个线程。adb logcat -v threadtime > F:\log\thread.log
java方法:
newJniThread创建线程,setJniEnv初始化 jvm 和 jobject。
c代码:
两个全局变量
下面是创建的线程执行的函数
怎么看创建的线程呢?
adb logcat -v threadtime >D:log\thread.log命令得到log日志
“new thread”还是在主线程中 后来的线程id 17043 、17044都变了
上面的代码j_obj会引起内存泄漏,解决办法:在写一个native方法,在方法里释放这个全局引用(*env)->DeleteGlobalRef(env,j_obj),然后在activity的onDestory方法中调用这个native方法。