安卓小白之ndk编译打包成so供第三方调用
本文接着上篇文章内容来,主要讲述如何生成so库以及将类编译成jar包供第三方调用
安卓小白之ndk入门篇
配置
在build.gradle中
externalNativeBuild {
cmake {
cppFlags ""
//在这里指定需要生成什么类型的so
abiFilters "armeabi","armeabi-v7a"
}
}
在CMakeList文件中可以指定生成的so库的名称
add_library( # Sets the name of the library.
native-lib #修改此处
# Sets the library as a shared library.
SHARED
# Provides a relative path to your source file(s).
src/main/cpp/native-lib.cpp)
.....
target_link_libraries( # Specifies the target library.
native-lib #修改此处
# Links the target library to the log library
# included in the NDK.
${log-lib})
注意add_library
的名称和target_link_libraries
的名称必须相同
还有另外一种配置
defaultConfig{
ndk{
//指定生成的so库名称
moduleName "native"
//在这里指定需要生成什么类型的so
abiFilters "armeabi","armeabi-v7a"
}
}
可以直接在gradle中都配置好,不需要更改CMakeList
配置好后,点击Build
->Make Project
等待编译完成,如果你是通过第一种方法配置的,那么生成的so库位于app
->build
->intermediates
->cmake
->debug
->obj
目录下,如果是第二种方法,那么so库位于app
->build
->intermediates
->ndk
->debug
->obj
目录下。
生成jar包
看cpp文件里的方法就知道,方法声明都是与包名绑定的,那我要是给第三方app调用呢,总不能让它们用我们的包名吧,所以,我们就需要将我们的类都打成jar包供第三方间接调用,让第三方调用我们的类,我们的类来调用c代码,这样问题就解决啦。
1.要编译jar包,前提条件是你的这个项目必须是一个类库,也就是library,在app gradle
的最上面查看是否为apply plugin: 'com.android.library'
2.在build defaultConfig目录里加入以下代码:
task makeJar(type: Copy) {
//删除存在的
delete 'build/libs/jar.jar'
//设置拷贝的文件
from('build/intermediates/bundles/debug/')
//打进jar包后的文件目录
into('build/libs/')
//将classes.jar放入build/libs/目录下
//include ,exclude参数来设置过滤
//(我们只关心classes.jar这个文件)
include('classes.jar')
//重命名
rename ('classes.jar', 'jar.jar')
}
makeJar.dependsOn(build)
看注释,名字任意取,然后再Terminal中执行命令gradlew makeJar
, Terminal在AndroidStudio的左下角。
执行完,提示BUILD SUCCESSFUL后,在以下路径可以找到生成的jar包:
app/build/intermediates/bundles/debug/
调用
jar包和so库都生成好后,就可以使用了,将jar包放入app->libs目录,将so库放入app->src->main->jinLibs
s
根据你的需要在jinLibs中建立不同的cpu架构目录,一般创建一个armeabi和armeabi-v7a就可以了,注意不要写错,同时注意在gradle中申明你需要的架构类型
ndk{
//在这里指定需要生成什么类型的so
abiFilters "armeabi","armeabi-v7a"
}
都配置好后,重新Sync Project下项目,就能愉快的使用了。
在使用中,有同学经常会遇到so库文件找不到的问题,这时你就需要检查你的架构目录下的so的数量是不是相同的,千万不要出现单个目录没有的情况。这是因为so库都是向下兼容,假如一台v7架构的手机,你把两个需要使用的so库都放到armeabi目录中,没问题,因为系统在v7中找不到会去armeabi目录找。但是如果你将一个so放到armeabi和armeabi-v7a,另一个只放了到了armeabi,当一个v7架构的手机使用的时候,如果发现armeabi-v7中没有是不会去armeabi中找的,因为另一个so也放到了armeabi-v7目录下,代表你兼容了armeabi-v7的情况,此时是不会向下兼容的。所以,解决方案就是要么删除v7里的so,要么把缺少的so补齐。