NDK产生的背景

Android平台从诞生起,就已经支持C、C++开发。众所周知,Android的SDK基于Java实现,这意味着基于Android SDK进行开发的第三方应用都必须使用Java语言。但这并不等同于“第三方应用只能使用Java”。在Android SDK首次发布时,Google就宣称其虚拟机Dalvik支持JNI编程方式,也就是第三方应用完全可以通过JNI调用自己的C动态库,即在Android平台上,“Java+C”的编程方式是一直都可以实现的。

不过,Google也表示,使用原生SDK编程相比Dalvik虚拟机也有一些劣势,Android SDK文档里,找不到任何JNI方面的帮助。即使第三方应用开发者使用JNI完成了自己的C动态链接库(so)开发,但是so如何和应用程序一起打包成apk并发布?这里面也存在技术障碍。比如程序更加复杂,兼容性难以保障,无法访问Framework API,Debug难度更大等。开发者需要自行斟酌使用。

于是NDK就应运而生了。NDK全称是Native Development Kit。

NDK的发布,使“Java+C”的开发方式终于转正,成为官方支持的开发方式。NDK将是Android平台支持C开发的开端。

为什么使用NDK

  1. 代码的保护。由于apk的java层代码很容易被反编译,而C/C++库反汇难度较大。
  2. 可以方便地使用现存的开源库。大部分现存的开源库都是用C/C++代码编写的。
  3. 提高程序的执行效率。将要求高性能的应用逻辑使用C开发,从而提高应用程序的执行效率。
  4. 便于移植。用C/C++写得库可以方便在其他的嵌入式平台上再次使用。

NDK简介

  1. NDK是一系列工具的集合
    NDK提供了一系列的工具,帮助开发者快速开发C(或C++)的动态库,并能自动将so和java应用一起打包成apk。这些工具对开发者的帮助是巨大的。
    NDK集成了交叉编译器,并提供了相应的mk文件隔离CPU、平台、ABI等差异,开发人员只需要简单修改mk文件(指出“哪些文件需要编译”、“编译特性要求”等),就可以创建出so。
    NDK可以自动地将so和Java应用一起打包,极大地减轻了开发人员的打包工作。

  2. NDK提供了一份稳定、功能有限的API头文件声明
    Google明确声明该API是稳定的,在后续所有版本中都稳定支持当前发布的API。从该版本的NDK中看出,这些API支持的功能非常有限,包含有:C标准库(libc)、标准数学库(libm)、压缩库(libz)、Log库(liblog)。

JNI简介

Java Native Interface (JNI)标准是java平台的一部分,它允许Java代码和其他语言写的代码进行交互。JNI 是本地编程接口,它使得在 Java 虚拟机 (VM) 内部运行的 Java 代码能够与用其它编程语言(如 C、C++ 和汇编语言)编写的应用程序和库进行交互操作。

NDK使用

在普通的Android工程中新建JniUtils类实现native方法
代码如下:

1
2
3
public class JniUtils {
public static native String getStringFormC();
}

然后clean project 再rebuild project 生成class文件,

这时候打开如下图的文件夹看是否生成了classes文件夹,没有生成请重新来过。

再打开Terminal输入指令
cd app/build/intermediates/classes/debug
然后再输入指令
javah -jni com.lin.astra.test.ndk_jni.JniUtils
注意 这里javah -jni后面跟的是JniUtils类的全路径,如果javah报不存在之类的,是你的java环境没有配置好。
这时候打开classes/debug下面的文件发现多了一个文件
com_lin_astra_test_ndk_jni_JniUtils.h

然后在src/main下新建文件夹jni,把生成的.h文件复制或者剪切到jni文件夹下面去,新建一个c类随便取一个名字,添加代码如下

1
2
3
4
5
#include "com_lin_astra_test_ndk_jni_JniUtils.h"
JNIEXPORT jstring JNICALL Java_com_lin_astra_test_ndk_1jni_JniUtils_getStringFormC
(JNIEnv \*env, jobject obj){
return (*env)->NewStringUTF(env, "这里是来自c的string");
}

在gradle.properties文件末尾添加android.useDeprecatedNdk=true
然后在app文件下得build.gradle ->defaultConfig括号内添加如下代码

1
2
3
static {
System.loadLibrary("NdkJniDemo");//之前在build.gradle里面设置的so名字,必须一致
}

然后简单调用就行了,MainActivity代码如下

1
2
3
4
5
6
7
8
9
10
public class MainActivity extends AppCompatActivity {
TextView textView;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
textView = (TextView) findViewById(R.id.ndk_text);
textView.setText(JniUtils.getStringFormC());
}
}

运行结果: