android 中为什么实用NDK,网上一搜一大堆,在这原因不在赘述。

在Android SDK首次发布时,Google就宣称其虚拟机Dalvik是支持JNI编程方式的;也就是第三方的应用是可以调用自己公司的或者是其他C、C++动态库。

搭建平台:1.我这用的是MacBook,系统是OS X 10.11.5;

                  2.Android Studio 2.1.2

                  3.JDK1.8.0_77 x86_64

首先搭建NDK开发环境。启动Android Studio,在Android studio中,属性—>Appearance & Behaveor —> System Settings —> Android SDK选项(注意:windows在使用NDKr8版本之上的NDK编译版本,是不需要cygwin来模拟linux,进行交叉编译的)。如下图所示:

wKiom1eDD0PRukVCAAdaML6qL6E045.png

打开右边的SDK Tools下边的选项栏有一个NDK的选项,打上对勾,点击Apply按钮,接着会出现下载NDK文件的对话框,点击accept—>OK,会下载NDK文件,通常NDK会安装到SDK的目录下面。

下载安装完成之后配置NDK环境变量:

1.Windows 上是在计算机—》属性—》高级—》环境变量,在path上追加NDK的安装目录。

2.linux或者是MAC是在家目录的.bash_profile中,添加环境变量,例如export NDK_HOME=/Users/XXX/Android/SDK/ndk-bundle export PATH=$PATH:$SDK_HOME

OK.测试NDK是否安装配置成功,打开终端,输入ndk-build -version.回车

显示

wKiom1eDED2T1ElgAAKFzKJlX4w955.png

表示安装成功。

1.创建一个Android项目来测试一个Android的使用方式,在这命名为JNIDemo;创建项目的过程不再赘述。


2.在Andorid视图中建立jni文件夹,右键—>New—>Folder—>JNI Folder,确定。会创建一个jni目录。这个目录就是存放c源码的文件。(jni目录的创建,也可以在project视图中在main文件中创建一个jni目录。如果在Android视图中创建JNI Folder,再切换到Project视图,你会发现,main目录下有一个jni目录,两个本质上是一致的)。

3.配置需要CUP编译成的架构库(.so文件)。

defaultConfig {

    applicationId "com.zzh.jni"

    minSdkVersion 14

    targetSdkVersion 23

    versionCode 1

    versionName "1.0"

    ndk{

        moduleName "zzhJni”//c/c++编译成的库文件名称

        abiFilters "armeabi", "armeabi-v7a","arm64-v8a","x86", "x86_64"//需要适配哪几种类型的CPU架构。

    }

}

4.在gradle.properties(Project Properties)中添加android.useDeprecatedNdk=true;

    在local.properties中添加ndk.dir=/Users/XXX/Android/SDK/ndk-bundle(自己安装NDK的路径)

5.新建一个jni调用类NdkUtils写一个native方法。在Java中,Java调用c/c++中的程序,需要使用到native关键字表示Java中调用C/C++中的方法。

package zzh;
public class NdkUtils {
    public static native String getMapHeader();
}

6.点击Build—>Make Project,编译文件,编译后

wKioL1eDEUyQ4eCJAAEOMeVYKJM296.png

在终端,进入到debug目录,运行javah -jni zzh.NdkUtils回车,生成c的头文件,头文件的命名为“包名_类名(包名之间的“.”使用“_”分割开的)”。将此文件复制到jni目录中,打开可以看到一下内容

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class zzh_NdkUtils */

#ifndef _Included_zzh_NdkUtils
#define _Included_zzh_NdkUtils
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     zzh_NdkUtils
 * Method:    getDeviceIdFromNdk
 * Signature: ()Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_zzh_NdkUtils_getMapHeader
        (JNIEnv *env, jobject obj);
#ifdef __cplusplus
}
#endif
#endif

    · JNIEXPORT jstring JNICALL Java_zzh_NdkUtils_getMapHeader(JNIEnv *env, jobject obj); 就是我们之前所用native在NdkUtils.java声明的方法。这个只是C/C++的一个.h文件,方法的具体实现,还要一个.c。这个.c文件,自己创建就好。

方法实现如下

#include "zzh_NdkUtils.h"
JNIEXPORT jstring JNICALL Java_zzh_NdkUtils_getMapHeader
        (JNIEnv *env, jobject obj){
    return (*env) -> NewStringUTF(env, "string from c");
}

7.至此C语言,就写好了,怎么让c/c++生成库文件呢,先点击Build-clean Project(先clean是因为避免在编译的时候产生冲突),再make Project,在6步骤的图片中classes同级目录下有ndk目录,此目录下有生成好的库文件。

wKiom1eDEouAOpHxAABjINxzj1I509.png

复制arm*、x86*到jniLibs目录中,就像我们在做项目时,将第三方的.so文件放入到jniLibs目录一样。Android.mk复制到jni文件中。

8.最后一步,调用生成的库文件。

在NdkUtils.java中,添加

static{

System.loadLibrary(“zzhJni"); //这一行,表示要加载C语言库,zzhJni就是编译成的c库文件,也就是在第三步中build.gradle中配置的moduleName 的名字

}

这样就可以在Java代码中调用c/c++方法了。

示例:

在MainActivity.java中调用:

public class MainActivity extends Activity {
    private TextView mTextView;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mTextView = (TextView) findViewById(R.id.textView);
        findViewById(R.id.button).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                try {
                    String str = "" + NdkUtils.getMapHeader();
                    mTextView.setText(str);
                } catch (Exception ex) {
                    ex.printStackTrace();
                } finally {

                }
            }
        });

    }
}

布局文件:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="com.zzh.jni.MainActivity">

    <Button
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="调用C方法"
        android:id="@+id/button"
        android:layout_alignParentTop="true"
        android:layout_alignParentLeft="true"
        android:layout_alignParentStart="true"/>

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="New Text"
        android:id="@+id/textView"
        android:layout_centerVertical="true"
        android:layout_centerHorizontal="true"/>
</RelativeLayout>

运行截图:

wKioL1eDF3CA-BtWAAGn7_z7YA0877.png

文笔有限,写的不好,敬请原谅,如有错误,敬请指正