AndroidStudio如何配置NDK/JNI开发环境

参考文章:

http://www.th7.cn/Program/Android/201509/550864.shtml

http://www.open-open.com/lib/view/open1451917048573.html

http://blog.csdn.NET/sodino/article/details/41946607

http://www.codes51.com/article/detail_197383.html

 

        AndroidStudio没有像Eclipse那样的一键add native support,相对来说比较麻烦,下面就介绍在AndroidStudio中如何实现类似于Eclipse的add native support功能(也就是进行NDK开发的步骤)。

 

1、新建一个Android工程,这一步就不多说了;

2、在AndroidStudio中配置NDK路径,方法是:

(1)先下载NDK并安装(这句基本是废话);

(2)点菜单栏的File->ProjectStructure…->在打开的窗口中左侧选中SDKLocation->在右侧Android NDK Location中填入NDK目录所在路径,如下图所示:

  


3、编译生成.class文件,方法是:

点菜单栏的Build->Make Project,如下图所示;

        这时,在工程的app/build/intermediates下就会生成classes文件夹,打开classes目录下的debug目录就会看到以你的包名命名的各级文件夹,最里边文件夹下有你的Java类对应的.class文件;

4、确定你要引用本地方法的类:

        其实你也可以先生成jni目录,再去创建这个类,但是先Google显然建议先创建要引用C代码的Java类,因为AndroidStudio可以根据你在java类中定义的native方法的名称来自动生成.h头文件。

        比如你想在MainActivity中引用本地方法,那么你先用

static {

        System.loadLibrary("myNativeLib");

}

来声明本地代码库,然后定义几个natvie方法,比如

public native String getStringFromNative();

做完这些工作就可以进行下一步了。

5、使用javah命令行生成jni目录及对应的头文件:

        我用的是AndroidStudio 2.1.1,在主界面最下边就能找到Terminal,点一下就能打开系统的命令行工具,并且已经为你自动cd到当前工程所在目录,如下图所示:


如上图所示,打开命令行编辑工具后,cd到工程的src/main/java目录,输入

     javah -d ../jni 你的包名.引用本地方法的类的名称

javah意思是生成一个.h头文件,-d ../jni的意思是生成一个名字叫做jni的文件夹(directory),位置是在当前目录(src/main/java)的上一级目录(即src/main目录);比如我的工程下,这条命令是

     javah -d ../jni com.example.lixinyu.myapplication2.MainActivity

我直接用了MainActivity类做为调用JNI的Java类,你也可以写一个自己的类,之后就等吧,构建完成后就会在工程的src/main目录下生成一个jni目录,下边还包含.h头文件,如下图所示:

        如果你事先在java代码中定义了native方法,那么这里还会自动生成方法对应的.h头文件。当然,要运行javah命令的前提是你的电脑已经安装过java,并且已经配置了java环境变量。

        这里再多说几句,很多网友说要使用javah -d jni -classpath命令来构建jni目录并生成.h头文件,但是我亲测是不行的,会报bash: ../../build/intermediates/classes/debug:is a directory错误,大概是版本不同吧,感谢码蚁之家的网友提供的不一样的答案。

        至此,我们就基本完成了相当于Eclipse中的add native support功能了,接下来还要配置gradle和一大堆其它的配置文件;

 

6、配置build.gradle文件

这里的build.gradle是指app模块下的build.gradle,不是整个工程的build.gradle文件。在模块的build.gradle的defaultConfig下加入以下idk配置:

ndk {

      moduleName"myNativeLib"

      ldLibs "log""z""m"

      abiFilters "armeabi""armeabi-v7a""x86"

}

其中moduleName是随便写的,与将来在Java类中使用System.loadLobrary(“本地库名称”);以及生成的.so文件名称对应;

ldLibs是要用到的jni库,一般由google提供,比如上边引入的log库可以让我们在C代码中使用LogCat日志;

abiFilters指的是我们要生成哪些平台的so文件,这里生成arm平台和x86平台;

配置后的build.gradle文件如下图所示:

        至此build.gradle文件就配置完了,其实这一步有点像我们eclipse中配置.mk文件;

7、配置local.properties文件

        打开工程目录下的local.properties,感觉这一步是自动配置的,或者说在你一开始在AndroidStudio中指定NDK目录时已经自动生成了。我的AndroidStudio在打开local.properties已经有了

ndk.dir=/Develop/Android/android-ndk-r10e

这一行,所以就不用配了;

8、配置gradle.properties

        打开工程目录下的gradle.properties文件(注意不是build.gradle,而是gradle.properties),在文件的最后一行加入

android.useDeprecatedNdk=true

这句的作用是允许我们使用已经过时的NDK版本,不知道AndroidStudio要求使用哪个版本的NDK才不会报错,总之只要配置了这一句就可以使用比较旧的NDK版本了,我用的r10;

        至此我们在AndroidStudio中就完成了NDK环境的配置,接下来就可以写Native代码了;

9、写一个.c文件测试一下是否运行正常

(1)在我们之前生成src/main/jni目录下新建一个.c文件,方法是在jni文件夹上点鼠标右键,选择New->C/C++ Source File,然后在弹出的对话框中填入.c或.cpp文件的文件名就可以了,比如说mail.c,名字可以随便起;

(2)在main.c中随便写一段JNI代码,比如如下所示的一段代码:

/* DO NOT EDIT THIS FILE - it is machine generated */

#include <jni.h>

#include <android/log.h>

 

#ifndef LOG_TAG

#define LOG_TAG"ANDROID_LAB"

#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR,LOG_TAG,__VA_ARGS__)

#endif

 

/* Header for class lab_sodino_jnitest_MainActivity */

 

#ifndef _Included_com_example_lixinyu_myapplication2_MainActivity

#define _Included_com_example_lixinyu_myapplication2_MainActivity

#ifdef __cplusplus

extern "C" {

#endif

/*

 * Class:lab_sodino_jnitest_MainActivity

 * Method: getStringFromNative

 * Signature: ()Ljava/lang/String;

 */

JNIEXPORT jstringJNICALL Java_com_example_lixinyu_myapplication2_MainActivity_getStringFromNative

  (JNIEnv *env, jobject jObj){

      LOGE("log string from ndk.");

      return (*env)->NewStringUTF(env,"HelloFrom JNI!");

  }

 

#ifdef __cplusplus

}

#endif

#endif

JNI基础就不多说了;

(3)在Java中调用:

比如我在MainActivity中加入以下代码:

package com.example.lixinyu.myapplication2;

 

import android.os.Bundle;

import android.support.v4.app.FragmentActivity;

import android.widget.TextView;

 

public class MainActivityextends FragmentActivity {

   

    private TextViewmTextView;

   

    static {

        System.loadLibrary("myNativeLib");

    }

   

    publicnative String getStringFromNative();

 

    @Override

    protected voidonCreate(BundlesavedInstanceState) {

        super.onCreate(savedInstanceState);

        setContentView(R.layout.activity_main);

 

        mTextView= (TextView) findViewById(R.id.main_textview);

        mTextView.setText(getStringFromNative()+"");

    }

}

       至此NDK/JNI代码就算是完成了,可以再次点菜单栏的Build->Make Project编译一下工程,看看在工程目录下app/build/intermediates/ndk/debug/lib目录下是否成功生成了相应的.so文件,如果以上配置都正确的话这里是会生成对应平台的.so文件的,debug目录下还自动生成了Android.mk文件,这个确实还不错,相当于我们用build.gradle文件的配置替代了Android.mk文件的配置,运行一下,正常的话是可以运行的,至此就算是在AndroidStudio中配置了NDK/JNI开发环境。

posted @ 2016-12-09 10:29  brave-sailor  阅读(378)  评论(0编辑  收藏  举报