2016年4月26日
网上有很多Android ndk搭建的教程,大多数都是用的gradle实验包。这对一个已经开发很久的工程加入jni开发,无疑加入很多的工作量。
这篇教程想比上一种方法略麻烦,但是基本不用改动多少gradle。废话不多说,我们开始。
[TOC]
准备工具
AndroidNdk开发需要ndk开发工具,好在现在的ndk不需要交叉编译了,所以能省下不少的事儿。Android Studio很良心,给我Android NDK下载的选项,也很方。但是我在安装过程中,出现了解压不了的bug。所以我还是推荐去官网下载,Android NDK。(自备梯子)
自己下载需要配置,Android Studio里面下载就不用了。下载之后解压,然后在Android Studio里面配置一下就好了。
然后配置ndk根目录到Path环境变量,这一会我们编译要用到。
关于路径问题
Android Studio默认c源码路径存放在/src/main/jni
下面,而编译出来的so文件放到jniLibs
下面,也就是说如果文件放到这两个目录里面你并不需要配置任何Gradle信息。当然,若果你有其他存放路径,请指定Gradle信息:
android{
...
sourceSets {
main {
jni.srcDirs = ['src/main/jni']
jniLibs.srcDirs = ['src/main/jniLibs']
}
}
...
}
编写native代码
生成头文件
private native String getString();
**然后编译一下。到java
路径下执行javah [类的全路径名]
比如
javah cn.liucl.ndkdemo.MainActivity
这个命令是生成头文件,有了头文件就可以写对应的方法实现,这个javah命令格式如下
-classpath <路径> 用于装入类的路径
-d <目录> 输出目录
-jni 生成 JNI样式的头文件(默认)
如果你有以上提示,那么一肯定没有编译,ctrl+f9一下。
我的头文件内容如下:(这是系统生成的)
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class cn_liucl_ndkdemo_MainActivity */
#ifndef _Included_cn_liucl_ndkdemo_MainActivity
#define _Included_cn_liucl_ndkdemo_MainActivity
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: cn_liucl_ndkdemo_MainActivity
* Method: getString
* Signature: ()Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_cn_liucl_ndkdemo_MainActivity_getString
(JNIEnv *, jobject);
#ifdef __cplusplus
}
#endif
#endif
JNIEXPORT jstring JNICALL Java_cn_liucl_ndkdemo_MainActivity_getString (JNIEnv *, jobject);
就是我们要实现的函数。
注意:你的包名和我的不一样。
Android.mk
然后把这个头文件拷贝到jni目录下吧,毕竟代码是写在这里的。**然后在jni路径下新建Android.mk
这是用来告知NDK Build 系统关于Source的信息,说白了就是用来解决依赖问题的。Android.mk写上去:
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := hello-jni #So文件loadLibrary()时候,指定它
LOCAL_SRC_FILES := hello-jni.c #你的C文件
include $(BUILD_SHARED_LIBRARY)
编写c
在jni目录下新建实现头文件的c文件。
#include <string.h>
#include <jni.h>
JNIEXPORT jstring JNICALL Java_cn_liucl_ndkdemo_MainActivity_getString
(JNIEnv * env, jobject obj)
{
return (*env)->NewStringUTF(env, "Hello from JNI !");
}
别忘了那两个include就可以了。
编译
build一下,会出现这个错误
我不知道为什么会这样,但的确要配置一下。
在你的项目根目录有个
gradle.properties
文件,加入android.useDeprecatedNdk=true
就可以了。
编译c代码,到jni路径下面执行ndk-build
,然后把生成的so文件拷贝到jniLibs
下面的armeabi
里面。默认生成这个版本。
运行
补全java代码,在调用native方法之前需要load进来so文件。注意,loadLibrary的内容一定要以Android.mk里面的LOCAL_MODULE为准。
public class MainActivity extends AppCompatActivity {
public static final String TAG = MainActivity.class.getSimpleName();
static {
System.loadLibrary("hello-jni");
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Log.i(TAG, "onCreate: " + getString());
}
private native String getString();
}
总结:以javah
来生成头文件,以ndk-build
来编译文件。虽然麻烦了一点,但是其实这个都可以写成脚本。避免gradle的大规模改动,引起项目不确定性。
参考:
Android.mk: http://www.cnblogs.com/wainiwann/p/3837936.html