Android Stuido NDK开发环境搭建

2016-04-30  本文已影响308人  liucloo

2016年4月26日

网上有很多Android ndk搭建的教程,大多数都是用的gradle实验包。这对一个已经开发很久的工程加入jni开发,无疑加入很多的工作量。
这篇教程想比上一种方法略麻烦,但是基本不用改动多少gradle。废话不多说,我们开始。

[TOC]

准备工具

AndroidNdk开发需要ndk开发工具,好在现在的ndk不需要交叉编译了,所以能省下不少的事儿。Android Studio很良心,给我Android NDK下载的选项,也很方。但是我在安装过程中,出现了解压不了的bug。所以我还是推荐去官网下载,Android NDK。(自备梯子

1.png-89.7kB1.png-89.7kB

自己下载需要配置,Android Studio里面下载就不用了。下载之后解压,然后在Android Studio里面配置一下就好了。

2.png-86.5kB2.png-86.5kB

然后配置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 [类的全路径名]

5.png-18.6kB5.png-18.6kB
比如javah cn.liucl.ndkdemo.MainActivity这个命令是生成头文件,有了头文件就可以写对应的方法实现,这个javah命令格式如下
-classpath <路径> 用于装入类的路径
-d <目录> 输出目录
-jni 生成 JNI样式的头文件(默认)
4.png-5.6kB4.png-5.6kB

如果你有以上提示,那么一肯定没有编译,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一下,会出现这个错误

3.png-24.5kB3.png-24.5kB
我不知道为什么会这样,但的确要配置一下。
在你的项目根目录有个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

上一篇 下一篇

猜你喜欢

热点阅读