NDK(Native Development Kit),是用于在 Android 应用中嵌套本地代码的工具集,现在 Android 开发最常用的工具就是 Android Studio 了,笔者也是刚刚接触 NDK 开发,用了一天的时间,踩了很多坑,最终跑通了自己的第一个 NDK 程序,话不多说,现在开始。
第一步:给 Android Studio 配置 NDK:
选择 SDK Manager
选择 SDK Tools
勾选 NDK 和一个 LLDB 版本,然后点击 Apply
接下来等待下载安装即可,下载完成后,右键点击工程 -- Open Module Setting -- SDK Location
然后选择 Android NDK location,点击 Select default NDK,然后点击 OK,就配置好了,然后开始写程序了
从第二步开始就是相当于属于 JNI(Java Native Interface | Java 本地接口) 开发的基础流程,因为 NDK 是基于 JNI 的。
第二步:创建一个 JNITest.java 的一个类:
public class JNITest {
//创建一个 native 方法
public native static String get();
}
注意:如果你创建的 get() 方法是红色的,并且有这样的提示:Cannot resolve corresponding JNI function Java_com_example_akon_jnitest_JNITest_get more..
解决办法:Android Studio --Preferences -- Plugins
搜索 NDK,然后将 Android NDK Support 后面的勾去掉,点击 OK,然后 Restart。当然,如果你没有这种情况就不用管。第三步:创建 C 语言文件,创建 so 库
点击 Make Project,生成 JNITest.class 文件
使用 Project 方式查看当前项目,在当前目录下可以看到你的 JNITest.class 文件
打开 Android Studio 的 Terminal 到 移动到 app/src/main 目录下
使用命令创建 .h 的头文件
javah -d jni -classpath F:\workSpace\Android\JNITest\app\build\intermediates\classes\debug (注意这里有空格分开) com.example.akon.jnitest.JNITest
javah:生成头文件
-d jni:当前目录下创建一个 jni 文件夹
-classpath .../debug
com.example.akon.jnitest.JNITest 是 JNITest 文件的包名加上字节码文件的名称
注意:这个 debug 文件目录可能太长,输入麻烦,我们可以找到 debug 文件夹,右键 copy path,复制文件目录即可
现在我们可以看到 app/src/main 目录下有一个 jni 文件夹,里面有一个 com_example_akon_jnitest_JNITest.h 的头文件,就是我们生成的头文件,头文件命名也是按照包名加字节码名的规范,以下划线连接。
然后在 jni 目录下创建一个 c/c++ resource 文件 test.c,要选择 c 为后缀:
#include<jni.h>
#include<stdio.h>
//导入我们创建的头文件
#include "com_example_akon_jnitest_JNITest.h"
JNIEXPORT jstring JNICALL Java_com_example_akon_jnitest_JNITest_get
(JNIEnv *env, jclass jclass){
//返回一个字符串
return (*env)->NewStringUTF(env,"This is my first NDK Application");
}
下面的这个方法使我们从头文件中复制过来的
然后修改了参数,给了返回值。
然后我们在 jni 目录下创建两个 .mk 文件:
一个 Android.mk:
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := JNITest
LOCAL_SRC_FILES := test.c
include $(BUILD_SHARED_LIBRARY)
一个是 Application.mk:
APP_ABI := all
然后我们需要生成 so 库:
打开 Terminal,到 app/src/main/jni 目录下,使用 ndk-build 命令生成 so 库:
注意:如果提示ndk-build: command not found 请执行以下操作:
1. 启动终端Terminal
2. 进入当前用户的home目录
输入cd ~
3. 创建.bash_profile
输入touch .bash_profile
4. 编辑.bash_profile文件
输入open -e .bash_profile
#路径是 你安装ndk 和 sdk的路径 ,需要自行调整.
export ANDROID_NDK_ROOT=/Users/lvxiangan/ProgramFiles/AndroidStudio/Sdk/ndk-bundle
export ANDROID_SDK_ROOT=/Users/lvxiangan/ProgramFiles/AndroidStudio/SDK
export PATH=$PATH:$ANDROID_SDK_ROOT
export PATH=$PATH:$ANDROID_NDK_ROOT
可以参考下图:
5. 保存文件,关闭.bash_profile
6. 更新刚配置的环境变量
输入source .bash_profile 也可以重新开启一个terminal #此时重新开启命令行, 当前配置也会生效.
7. 验证配置是否成功:命令行输入 ndk-build
接着打开 app/src/main/libs 就可以看见我们生成的 so 库了。
为了防止 so 库兼容错误,在 gradle.properties 最后一行添加:
android.useDeprecatedNdk=true
为了让项目能够找到我们的 so 库,在 build.gradle 文件夹的 android 下添加:
sourceSets {
main() {
jniLibs.srcDirs = ['src/main/libs']
jni.srcDirs = [] //屏蔽掉默认的jni编译生成过程
}
}
然后我们在 JNITest.java 中动态导入 so 库,不需要写 libJNITest,只用写 JNITest:
package com.example.akon.jnitest;
public class JNITest {
// 动态导入 so 库
static {
System.loadLibrary("JNITest");
}
//创建一个 native 方法
public native static String get();
}
然后我们在 MainActivity 中打印 JNITest 的 get() 方法获取到的 String 值:
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//打印信息
Log.e("Message",JNITest.get());
}
}
然后运行,查看日志,搜索 Message:
然后我们就查看到了我们获取的 String 了,“This is my first NDK Application”,就是我们在 C 语言文件中所返回的字符串。
好了,第一个 NDK 程序就写完了。