AndroidStudio如何引入so包

先说前提条件,我的AndroidStudio版本是2.2 Preview 3,版本是2.1的同学应该也是一样的。

然后说结论吧,有些同学可能赶着做项目:

结论:

so包应该放在相应模块(比如app模块)下的src目录下的main目录下的jniLibs目录。

注意是jniLibs,最后边有个s,不是jniLib,并且L要大写。如果你在src/main目录中看不到jniLibs目录,那你只需要自己建一个这个目录就可以了,然后把你的so包按编译平台分类拷贝进去就可以了。然后呢?然后就完了,就这样就可以了,因为系统默认就会去这个目录中找对应的so包。如下图所示:



接下来有兴趣的同学可以了解下为什么是这样。

也许有人说,so包应该跟jar包一样呗,放在对应模块的libs目录下,eclipse就是这样的。有这么简单吗?答案是:NO

原理:

其实在AndroidStudio中,我们之所以可以把jar包放在对应模块的libs目录下,比如app模块(也就是通常意义下的主工程了)的libs目录中,而不需要再去配置build.gradle文件,就可以正常使用这些jar包,是因为在AndroidStudio中新建项目时,系统已经在默认为我们配置好了gradle,如果仔细看模块下的build.gradle文件的话,一般会有这么一句:

dependencies {
    compile fileTree(dir: 'libs', include: ['*.jar'])
}

看到了吧,这句就是告诉gradle,我们的第三方jar包在libs目录下,如果没有这一行配置,那么理论上我们把jar包直接放在libs目录下其实是不行的,只不过新建项目时系统已经为我们配好了。这里libs是个相对路径,因为我们的build.gradle文件本身就在app模块下,那么这里的libs当然也就是指app模块下的libs,其实build.gradle文件中所有的路径都可以写相对路径,我们下边讲的so包的路径也是只写相对路径就可以了。

OK,言归正传,现在知道jar包的引入原理了,那么so包为什么要放在那么奇怪的目录下?能不能放在任意一个自定义的目录呢?当然是可以的。

其实在build.gradle中,默认会有一些这样的配置:

sourceSets {
    main {
        jniLibs.srcDirs = ['src/main/jniLibs']
        aidl.srcDirs = ['src/main/aidl']
    }
}

这些配置在你新建工程之后,在build.gradle文件中默认是没有的,因为默认就是这样的,当然也就不写也可以,当然你写出来也没错。意思是so包就去src/main/jniLibs目录下找,aidl文件默认就到src/main/aidl目录下找。。。当然还有许多其它的配置,你还可以配置你的AndroidManifest.xml文件在哪里,还可以配置你的java代码在哪里,如果你不配的话都会有一个默认值,这里只是以jniLibs和aidl为例而已。

看到这里你应该就明白了,为什么我们把so包直接放到src目录下的main目录下的jinLibs目录就可以了,而不需要配置gradle文件了,因为系统默认就会到这个目录下找,如果你想把so包放在一个你喜欢的地方,比如直接放在app目录下的myJniLibs目录下,那你只需要把上面内容改成:

sourceSets {
    main {
        jniLibs.srcDirs = ['myJniLibs']
        aidl.srcDirs = ['src/main/aidl']
    }
}

这样就可以了。明白了原理,似乎一切都明了了,同理,如果不想把aidl文件放在默认目录下,也可以改aidl.srcDirs的值,如果不想把java代码放在默认目录,就可以改java.srcDirs的值,如果不想把资源文件放在默认目录下,就可以改res.srcDirs的值。。。还有几个,同学们慢慢去研究吧,如果你想把目录结构改的跟eclipse下的工程一样,就改这里的值就可以,不一一细说了。

### 如何在 Android Studio 中添加并配置外部 .so 文件 #### 设置 SO 文件路径 为了使应用程序能够找到所需的 `.so` 库,在 `build.gradle` (Module: app) 文件中的 `android` 块下指定源集的 jniLibs 目录。默认情况下,Android Studio 的 so 文件放置位置为 `app/src/main/jniLibs/armeabi` 或者其他 ABI 版本对应的文件夹内[^1]。 ```gradle sourceSets { main { jniLibs.srcDirs = ['libs'] } } ``` 上述代码片段会告诉构建工具去名为 `libs` 的目录里查找本地库而不是默认的 `src/main/jniLibs` 路径。 #### 配置 Gradle 构建脚本 对于那些已经编译好的第三方共享库(即`.so`),可以直接复制到项目的适当ABI子目录中,并通过修改模块级 `build.gradle` 来确保这些库被正确打进 APK 里面: ```gradle android { ... defaultConfig { externalNativeBuild { cmake { cppFlags "" } } ndk { abiFilters 'armeabi-v7a', 'arm64-v8a' // 只含特定架构版本 } } buildTypes { release { minifyEnabled false proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro' } } externalNativeBuild { cmake { path "CMakeLists.txt" } } } ``` 这段配置指定了哪些 CPU 架构应该被打以及告知 CMake 使用哪个清单文件来进行链接操作[^3]。 #### 将预编译的 .so 文件加入工程 把预先编译得到的 `.so` 动态链接库放入合适的位置——通常是按照不同平台分别存放在如下结构下的相应文件夹中: - `app/src/main/jniLibs/armeabi/libexample.so` - `app/src/main/jniLibs/armeabi-v7a/libexample.so` - `app/src/main/jniLibs/arm64-v8a/libexample.so` - ... 这样做之后,当应用安装时就会自动把这些二进制资源嵌入最终的应用程序(APK)[^4]。 #### 加载动态库 最后一步是在 Java/Kotlin 层面加载这个 native library 。这通常发生在 Application 类或者 Activity 生命周期早期阶段,比如 onCreate 方法内部: ```java static { System.loadLibrary("example"); } ``` 这里假设所导入的 `.so` 文件名称叫做 libexample.so ,去掉前缀 “lib” 和后缀“.so”,只留下中间部分作为参数传递给 loadLibrary 函数调用[^2]。
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值