LJ的Blog

学海无涯苦做舟

0%

Android-NDK学习(1)

写在前面

最近这两天非常烦躁,移植A项目的功能到B项目里,中间牵扯到的种种资源什么的就不说了,有耐心慢慢剥离就行。中间有一个so库一直加载失败,我真的是日了狗了,尝试了各种办法还是不行。最后用了他的sample里的东西作为module导入才解决,但是我觉得这并不算特别完美的解决办法,唉,没办法,对ndk了解的还是太少,从现在开始做相应的了解。

第一个总是Hello World

Android Studio在现在的版本可以非常方便的生成一个带jni的项目,只要在创建的时候勾选一个include c++ 那玩意就行了。不过对于我这种刚接触的人来说,还是要自己折腾一下。

首先是配置ndk开发环境,其实也没什么要配置的,一顿下载就ok了。

ndk

这俩玩意勾选上,等他慢慢下吧。下完之后设置一下ndk的路径:

路径

配置好了路径之后,我们创建一个普通的android项目,恩,不勾选c++那个选项,自己慢慢折腾。项目创建完毕之后,正确的项目目录是这样的:

目录

当然你并没有jni这个目录,可以自己创建一个:

创建jni目录

创建完之后在MainActivity中声明一个native方法:

1
public static native String getHelloWorldStr();

选中这个函数按alt+enter(mac为option + return,即android studio自动提示),点击第一个选项:

创建源文件

生成.c文件源码如下:

1
2
3
4
5
6
#include <jni.h>

JNIEXPORT jstring JNICALL
Java_com_xiasuhuei321_studyforndk2_MainActivity_getHelloWorldStr(JNIEnv *env, jclass type) {
return (*env)->NewStringUTF(env, "Hello World!");
}

其中”Hello World!”字符串是我自己根据需要改成那样的,现在不去深究c语言代码的意思,只管走通整个流程。用过jni的人应该知道,用jni需要根据用户自己的class文件生成一个.h头文件,在Android中也是需要这个头文件的。在android studio中怎么生成这个头文件我也是头疼了一会,因为也没去了解过as把class文件输出到哪个目录了,好在后来找到了一个生成的方法。首先打开android studio中自带的Terminal,打开就已经在项目根目录中了,省的自己一个一个的去切换了。windows用户可以用dir查看目录下有哪些文件和文件夹,mac、Linux用户可以用ls查看。进入项目的/src/main/java 目录下,输入以下命令:

1
javah -d ../jni com.xiasuhuei321.studyforndk2.MainActivity

这样就在jni目录下生成了一个.h头文件

头文件

在c文件中加入这个头文件:

1
#include <com_xiasuhuei321_studyforndk2_MainActivity.h>

接着在MainActivity中加载我们的库:

1
2
3
static {
System.loadLibrary("JNI_ANDROID_TEST");
}

static代码块可以保证我们的代码在实例创建之前就被执行。在Android中生成的so一般都是有lib前缀的,比如我上面用的这个名字,生成的so就会是libJNI_ANDROID_TEST.so这个名字。最后在项目app的build.gradle中的defaultConfig中加入如下代码:

1
2
3
ndk{
moduleName "JNI_ANDROID_TEST"
}

这句话指定编译生成so的名字,至于通过这种方式生成的so目录位置,我找了一下,在这个位置:/app/build/intermediates/ndk/debug/lib/

lib中有七个abi目录,可以根据自己的需求指定abiFilter,不用生成那么多so。如果你对Android中的CPU架构和ABI不是非常清楚,可以自行去查阅资料。

java调用native方法代码:

1
2
tv_text = (TextView) findViewById(R.id.tv_text);
tv_text.setText(getHelloWorldStr());

最后免不了看一下结果:

结果
textview默认文字是C语言调用失败,这么一来便足以证明此次流程通了。

这里我感觉这头文件是不是也可以通过gradle配置来生成,毕竟每次都这么来一遭也挺麻烦的,后续如果我发现可以,就回来补充一下,没有就算了。