Android Studio下NDK开发流程

以前Android NDK开发需要在Eclipse或源码环境下,简历并配置Android.mk和Application.mk,并且还要通过java命令生产.h头文件才能编译生成so库,相当麻烦。随着AS作为官方Android开发工具,现在准备在AS上开发JNI应用,发现在AS上编译NDK非常方便,本文将介绍如何在Android Studio上实现NDK开发。


1、简介

JNI

JNI 是Java Native Inteface的缩写,是Java中定义的一种用于连接Java和C/C++接口的一种实现方式。

NDK

NDK 是 Native Developmentit的缩写,是Google在Android开发中提供的一套用于快速创建native工程的一个工具。 
使用这个工具可以很方便的编写和调试JNI的代码

Gradle

Gradle 是一个基于Apache Ant和Apache Maven概念的项目自动化建构工具。它使用一种基于Groovy的特定领域语言(DSL)来声明项目设置。
环境要求

  • Android Studio 2.2或更高版本
  • Gradle 2.2或更高版本
  • Android NDK r10e或更高 下载地址
  • Build Tools 19.0.0或更高

2、开始集成

a) 创建一个简单的Android项目

新建一个Android测试项目,命名为:JniDemo。 

b) 配置NDK路径

找到AS的状态栏,打开File->Project Structure,在Android NDK location栏选择离线NDK保存路径. 

c) 配置gradle.properties文件

JNI模块构建方式有3种: 

  • 手动ndk-build构建。该方式需要手动编写Android.mk和Application.mk文件,然后手动调用ndk-build命令生成so文件,属于早期NDK开发方式,过程比较繁琐。 
  • 自动ndk-build构建。使用ndkCompile工具自动构建so文件,需要手动生成头文件。 
  • gradle-experimental方式全自动构建。 

我们使用最简单最方便的构建方式,所以使用第二种。找到工程中的gradle.properties文件,添加以下代码:

android.useDeprecatedNdk=true


d) 配置build.gradle文件

找到主module中的build.gradle文件,添加一下代码:

ndk{ 
    moduleName "JniCore"//编译生成so库的名字,注意不要lib和.so,和loadLibrary里面的参数一致
    ldLibs "log", "z", "m" //添加日志依赖库
    abiFilters "armeabi", "armeabi-v7a", "x86" //指定生成编译支持的平台so
}


e) JNI编码

在java同级创建一个jni目录,相关的C/C++源码放在该目录下。我们这里以实现MD5哈希摘要为例。 


f) JNI注册

目前JNI注册方式有两种:静态注册和动态注册。个人偏向于使用动态注册,一方面可以防止方法名hook,另外一方面省了写.h文件。 

JNI入口文件:main.c

/* DO NOT EDIT THIS FILE - it is machine generated */
        #include <jni.h>
        #include <android/log.h>
        #include <stdlib.h>
        #include <stdio.h>

        #include "md5.h"

        #define LOG_ENABLE
        #define  LOG_TAG "JNI_LOG"
        #define  LOGI(...)  __android_log_print(ANDROID_LOG_INFO, LOG_TAG, __VA_ARGS__)
        #define  LOGE(...)  __android_log_print(ANDROID_LOG_ERROR, LOG_TAG, __VA_ARGS__)

        //指定注册的类,需要跟Java中native方法所在的包名一致
        #define JNIREG_CLASS "com/luoxudong/jnidemo/natives/JniCore"

        JNIEXPORT jstring JNICALL n_md5(JNIEnv * env, jobject jObj, jstring plaintext)
        {
            if (plaintext == NULL)
           {
                return NULL;
           }
           const char *pPlaintext = (*env)->GetStringUTFChars(env, plaintext, 0);
           char cMD5[32 + 1] = {0};
           MD5(pPlaintext, strlen(pPlaintext), cMD5);
         (*env)->ReleaseStringUTFChars(env, plaintext, pPlaintext);
          jstring ret = (*env)->NewStringUTF(env, (char *)cMD5);
          return ret;
        }

        static JNINativeMethod method_table[] = {
               {"md5", "(Ljava/lang/String;)Ljava/lang/String;", (void *)n_md5}
        };

        static int registerNativeMethods(JNIEnv* env, const char* className, JNINativeMethod* gMethods, int numMethods)
        {
         jclass clazz;
          clazz = (*env)->FindClass(env, className);
           if (clazz == NULL)
          {
              return JNI_FALSE;
          }
          if ((*env)->RegisterNatives(env, clazz, gMethods, numMethods) < 0)
          {
             return JNI_FALSE;
         }
         return JNI_TRUE;
        }

        static int registerNatives(JNIEnv* env)
        {
           if (!registerNativeMethods(env, JNIREG_CLASS, method_table, sizeof(method_table) / sizeof(method_table[0])))
          {
              return JNI_FALSE;
          }
           return JNI_TRUE;
        }

        JNIEXPORT jint JNI_OnLoad(JavaVM* vm, void* reserved)
        {
          JNIEnv* env = NULL;
          jint result = -1;
         if ((*vm)->GetEnv(vm, (void **)&env, JNI_VERSION_1_4) != JNI_OK)
          {
              return -1;
          }
         if (!registerNatives(env)) //注册
          {
              return -1;
          }
         result = JNI_VERSION_1_4;

          return result;
        }

MD5头文件:md5.h

省略...

MD5原文件:md5.c

省略...

g) JAVA代码实现

创建类com.luoxudong.jnidemo.natives.JniCore.java文件,需要跟main.c中的”com/luoxudong/jnidemo/natives/JniCore”对应。 

native文件:JniCore.java

package com.luoxudong.jnidemo.natives;

       public class JniCore {
         static {
             System.loadLibrary("JniCore");
          }
          public static native String md5(String text);
       }

h) native方法调用

package com.luoxudong.jnidemo;

import android.app.Activity;
import android.os.Bundle;
import android.widget.TextView;

import com.luoxudong.jnidemo.natives.JniCore;

public class MainActivity extends Activity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
      super.onCreate(savedInstanceState);
      setContentView(R.layout.activity_main);
      TextView msgTxt = (TextView) findViewById(R.id.tv_msg);
      msgTxt.setText(JniCore.md5("test"));
    }
}

i) 生成so文件

编译工程,在app/build/intermediates/ndk/debug/lib下面可以看到生成的各平台so文件。 

j) 其他

源码下载地址:JniDemo
转载请注明出处:http://www.luoxudong.com/?p=139

发表回复

您的电子邮箱地址不会被公开。