Android JNI/NDK开发JNI实现C/C++与Android/JAVA相互调用

Posted Ruffian-痞子

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Android JNI/NDK开发JNI实现C/C++与Android/JAVA相互调用相关的知识,希望对你有一定的参考价值。

前言
一直觉得无论是从行业形式来看,还是从就业角度说,学习JNI/NDK开发是一条必经之路。很多说androidios几乎没有市场了,其实只能说现在开发APP的需求没那么多了,再加上之前培训机构疯狂向市场输出Android/IOS开发人员,导致市场供过于求,难免出现一职难求的现象。不过我不认为Android会死,虽然APP的需求少了,但是Android又不是只能做APP,除非你只会APP开发。哈哈

第一部分:环境配置和简单demo运行参考
Android JNI/NDK开发(一)NDK真的很难吗?

1.1JNI/NDK在Android studio 上基本配置

参考连接http://blog.csdn.net/krubo1/article/details/50547681

1.2Android Studio 定制快速生成Jni 头文件工具 NDK教程

参考连接http://blog.csdn.net/silver_R/article/details/48457077

备注1:
在jni文件下添加一个空的C文件 empty.c 能解决如下报错

Error:Execution failed for task ‘:app:compileDebugNdk’. 
NDK_PROJECT_PATH=null

备注2:
在 app > build.gradle 中配置ndk

        //ndk配置
        ndk 
            moduleName "RLibs"
            //so文件包名
            stl "stlport_static"
            //NDK中C++标准库、STL的配置
            //如未配置可能出现错误: fatal error: iostream: No such file or directory
            ldLibs "log", "z", "m"
            abiFilters "armeabi", "armeabi-v7a", "x86"
        

第二部分(使用Android studio实现):JNI实现C++与Java之间函数的相互调用

我们知道Java和C/C++是两种不同的语言,不能直接通信,所以需要通过中间桥梁JNI将他们连接起来。

JNI充当中间桥梁,那么它肯定存在很多函数可以将Java的基本数据类型转化为JNI通用的类型,相对于C/C++也是一样。下面给出一个

JNI中文学习文档http://www.cnblogs.com/jycboy/archive/2016/04/15/5396876.html#jialianbendi

给出一片文档就算了?那怎么可能啊,当时我在网上看到这个文档的时候我觉得JNI没问题了,但还是动手试了一下,不试不知道,动手一敲,一脸懵逼。那还是跟着看看Java层代码是如何跟C/C++代码交互的吧。

需求:
1.1:Java调用C++方法获取基本数据类型(int,double,string)
1.1:Java调用C++无返回值方法
2.1:C++调用Java非静态方法获取基本数据类型(int,double,string)
2.2:C++调用Java静态方法获取返回值
2.3:C++调用Java无返回值方法

OK,Java中写一个native工具类,定义native方法,以及Java中方法的实现

/**
 * native工具类
 *
 * @author ZhongDaFeng
 * @date 2017/5/3 14:48
 */
public class NativeUtils 

    public static Context mContext;

    static 
        System.loadLibrary("RLibs");
    

    //从native获取int类型数据
    public native int getIntFromNative(int javaInt);

    //从native获取double类型数据
    public native double getDoubleFromNative(double javaDouble);

    //从native获取string类型数据
    public native String getStringFromNative(String javaString);

    //从native获取ArrayList<String>类型数据
    public native ArrayList<String> getArrayListFromNative(ArrayList<String> javaList);

    //模拟native层调用安卓Toast
    public native void nativeCallAndroidToast();

    //模拟native层调用java基本类型信息
    public native String nativeCallJavaBaseMsg();


    //native获取int类型数据
    public int getIntFromJava(int nativeInt) 
        int javaInt = 100;
        return javaInt + nativeInt;
    

    //native获取double类型数据
    public double getDoubleFromJava(double nativeDouble) 
        double javaDouble = 100.0;
        return javaDouble + nativeDouble;
    

    //native获取string类型数据
    public String getStringFromJava(String nativeString) 
        String javaStr = " Java Static String : ";
        String split = " call ";
        return nativeString + split + javaStr ;
    

    //native获取ArrayList<String>类型数据
    public ArrayList<String> getArrayListFromJava(ArrayList<String> nativeList) 
        return null;
    

    //native 获取静态方法数据
    public static String getStringFromJavaWithStatic(String nativeString) 
        String javaStr = " java static string  ";
        String split = " && ";
        return javaStr + split + nativeString;
    

    // 提示信息,提供JNI调用
    public void showToast(String msg) 
        Toast.makeText(mContext, msg, Toast.LENGTH_SHORT).show();
    

根据这个类生成com_r_ndk_NativeUtils.h头文件(文章开头有说明如何生成)

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

#ifndef _Included_com_r_ndk_NativeUtils
#define _Included_com_r_ndk_NativeUtils
#ifdef __cplusplus
extern "C" 
#endif
/*
 * Class:     com_r_ndk_NativeUtils
 * Method:    getIntFromNative
 * Signature: (I)I
 */
JNIEXPORT jint JNICALL Java_com_r_ndk_NativeUtils_getIntFromNative
  (JNIEnv *, jobject, jint);

/*
 * Class:     com_r_ndk_NativeUtils
 * Method:    getDoubleFromNative
 * Signature: (D)D
 */
JNIEXPORT jdouble JNICALL Java_com_r_ndk_NativeUtils_getDoubleFromNative
  (JNIEnv *, jobject, jdouble);

/*
 * Class:     com_r_ndk_NativeUtils
 * Method:    getStringFromNative
 * Signature: (Ljava/lang/String;)Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_com_r_ndk_NativeUtils_getStringFromNative
  (JNIEnv *, jobject, jstring);

/*
 * Class:     com_r_ndk_NativeUtils
 * Method:    getArrayListFromNative
 * Signature: (Ljava/util/ArrayList;)Ljava/util/ArrayList;
 */
JNIEXPORT jobject JNICALL Java_com_r_ndk_NativeUtils_getArrayListFromNative
  (JNIEnv *, jobject, jobject);

/*
 * Class:     com_r_ndk_NativeUtils
 * Method:    nativeCallAndroidToast
 * Signature: ()V
 */
JNIEXPORT void JNICALL Java_com_r_ndk_NativeUtils_nativeCallAndroidToast
  (JNIEnv *, jobject);

/*
 * Class:     com_r_ndk_NativeUtils
 * Method:    nativeCallJavaBaseMsg
 * Signature: ()Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_com_r_ndk_NativeUtils_nativeCallJavaBaseMsg
  (JNIEnv *, jobject);

#ifdef __cplusplus

#endif
#endif

复制h文件,修改为C/C++文件(随便命名)实现业务逻辑

第一个需求:Java调用C++方法获取基本数据类型(int,double,string)

先编写C++逻辑

/*
 * Class:     com_r_ndk_NativeUtils
 * Method:    getIntFromNative
 * Signature: (I)I
 */
JNIEXPORT jint JNICALL Java_com_r_ndk_NativeUtils_getIntFromNative
  (JNIEnv * env, jobject obj, jint j_int)
   int result=50;
   result+=j_int;
   return result;
  

/*
 * Class:     com_r_ndk_NativeUtils
 * Method:    getDoubleFromNative
 * Signature: (D)D
 */
JNIEXPORT jdouble JNICALL Java_com_r_ndk_NativeUtils_getDoubleFromNative
  (JNIEnv * env, jobject obj, jdouble j_double)
    double result=50.0;
     result+=j_double;
     return result;
  

这两个函数,不难看出,JNI接受从Java传递过来的int /double类型的值,在本地代码(C++)中加上一定的值然后再返回给Java
jint/jdouble/jstring 等等是JNI的数据类型(第一篇博客中有说明),在C++中int/double类型可以直接转为对应的JNI类型使用。
所以到这里我们解决了一个问题:
从Java传递值过来,在生成h头文件的时候会直接转化为JNI对应的数据类型 int=>jint,而对于C/C++中int/double等类型可以直接转化为JNI的类型使用,那么JNI就将Java和C++这些基本的数据类型连接起来了。

这个不难,相对可恶的是string类型的数据,我们知道其实string不是Java的基本数据类型,而是封装的对象,所以这个string的处理比较麻烦一点。现在来分析一下string字符串处理的函数,代码:

/*
 * Class:     com_r_ndk_NativeUtils
 * Method:    getStringFromNative
 * Signature: (Ljava/lang/String;)Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_com_r_ndk_NativeUtils_getStringFromNative
  (JNIEnv * env, jobject obj, jstring j_string)

      const char *java_str_char = env->GetStringUTFChars(j_string, NULL);//获取java字符串
      if(java_str_char == NULL)
        return NULL;
      
      char const *native_char = " Native :";
      char const *split_char = " call ";
      std::string const& temp = std::string(java_str_char) +std::string(split_char) + std::string(native_char);
      char const *result = temp.c_str();
      env->ReleaseStringUTFChars( j_string, java_str_char);//释放
      return env->NewStringUTF(result);
  

对于int、double这类数据JNI和C++是可以直接转换使用,那么对于不能直接转换的怎么办呢?肯定会有一些函数的对吧,我们查看刚刚那个JNI中文文档
解读一下这个函数的代码(C++实现)
函数第1行获取Java的string类型数据转化为C++中的字符,JNI函数:env->GetStringUTFChars(j_string, NULL);
接着判断字符串是否==null,据了解,C/C++不会报空指针之类的错误,如果这里为空没有判断,下面代码还是使用会导致很严重问题,比如说你找错找的一脸懵逼,完全不知道哪里的问题,培养严谨的编程习惯,这些一定要判断,(获取接下来的示例中我可能没有写这些判断,不要学我,我只是demo快速写逻辑而已),接下来的代码就是在C++中创建字符,然后转成string类型,将Java传递过来的string字符串和C++的字符串拼接起来再返回出去。这里还要注意:env->ReleaseStringUTFChars( j_string, java_str_char);//释放资源方法,第一行代码获取字符串,这里释放资源,成对使用。
其实如果懂得C++编程的话这里还是很简单的,无非就是如何在C++中创建字符,传唤成为字符串,拼接起来。
C++部分的代码写完了,在Java中调用,代码:

                //模拟Java层调用native基本信息
                msg = mNativeUtils.getStringFromNative(" Java ");
                int nativeInt = mNativeUtils.getIntFromNative(33);
                double nativeDouble = mNativeUtils.getDoubleFromNative(33.33);
                String result = msg + " int: " + nativeInt + " double: " + nativeDouble;
                mtv_text.setText(result);

效果图

第二个需求:C++调用Java非静态方法获取基本数据类型(int,double,string)

直接看C++代码

/*
 * Class:     com_r_ndk_NativeUtils
 * Method:    nativeCallJavaBaseMsg
 * Signature: ()Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_com_r_ndk_NativeUtils_nativeCallJavaBaseMsg
   (JNIEnv * env, jobject obj)

    jclass j_class = env->FindClass("com/r/ndk/NativeUtils");
    jobject j_object = env->AllocObject(j_class);

    jmethodID j_method_id1 = env->GetMethodID(j_class, "getStringFromJava","(Ljava/lang/String;)Ljava/lang/String;");
    jmethodID j_method_id2 = env->GetMethodID(j_class, "getDoubleFromJava","(D)D");
    jmethodID j_method_id3 = env->GetMethodID(j_class, "getIntFromJava","(I)I");

    jstring j_str1 = (jstring) env->NewStringUTF(" Native ") ;
    jstring j_str2 = (jstring) env->CallObjectMethod(j_object, j_method_id1,j_str1);//从java层获取string
    jdouble j_double=(jdouble) env->CallDoubleMethod(j_object, j_method_id2,66.66);//从Java层获取double
    jint j_int=(jint) env->CallIntMethod(j_object, j_method_id3,66);//从Java层获取int

    string j_double_str=double2string(j_double);
    string j_int_str=int2string(j_int);
    string tip_int=" int: ";
    string tip_double=" double: ";

    const char * result_string_char = env->GetStringUTFChars(j_str2, NULL);//获取java字符串
    string result_number_str = tip_int + j_int_str + tip_double + j_double_str;//拼接数字
    string result_str = std::string(result_string_char)  + result_number_str;//拼接数字和字符串
    const char* result_char = result_str.c_str();//字符串转换为char
    env->ReleaseStringUTFChars( j_str2, result_string_char);//释放
    return env->NewStringUTF(result_char);
   

解析:在C++层获取Java数据,首先需要知道要调用的 方法 在哪个 参数 类型等
1.首先获取需要操作的类:env->FindClass(“com/r/ndk/NativeUtils”);
备注:填写类的绝对路径,将 . 换成 /
2.找到需要调用的方法的ID:env->GetMethodID(j_class, “getStringFromJava”,”(Ljava/lang/String;)Ljava/lang/String;”);
备注:第一个参数 类名,第二个参数方法名,第三个参数签名类型
JNI使用Java虚拟机的类型签名表述。下表列出了这些类型签名:

例如,Java方法:

long f (int n, String s, int[] arr);

具有以下类型签名:

(ILjava/lang/String;[I)J

特别注意一点:全限定的类后面一定要加 ; 否则开发中遇到运行报错,一脸懵逼不知到什么问题
获取方法ID可以查看JNI中文文档,基本上是:GetMethodID,对于调用静态方法:GetStaticMethodID

3.调用方法:env->CallObjectMethod(j_object, j_method_id1,j_str1);
备注:第一个参数对象,第二个参数方法ID,第三个参数传递的数值(可传多个参数,多种类型的值)
调用方法原型 CallMethod(参数,参数,参数…)type:object,int,double,void ….
静态方法调用:CallStaticObjectMethod(j_class, j_method_id2 ,j_string1); 不同于普通方法,第一个参数需要传class而不是object

这个C++函数主要是介绍这三个方法的使用,其他的逻辑就是,获取到数值转化为string拼接然后返回,很简单的啦。其中有两个方法的代码没有贴上来,是int/double转化为C++字符串的代码,之后在完整的demo中会给出。
Java调用代码:

                //模拟native层调用Java基本数据类型
                msg = mNativeUtils.nativeCallJavaBaseMsg();
                mtv_text.setText(msg);

这个demo在C++层中获取Java的基本数据类型,简单处理一下,再返回给Java展示。

到这里基本上实现了C++和Java之间数据的相互调用和传递,接着再实现一下,当C++调用Java之后,Java在处理一下的方法:在C++中传递一个string字符串,Java接收之后Toast提示给用户。(顺便讲解一下C++调用Java静态方法的用例)
看C++代码

/*
 * Class:     com_r_ndk_NativeUtils
 * Method:    nativeCallAndroidToast
 * Signature: ()V
 */
JNIEXPORT void JNICALL Java_com_r_ndk_NativeUtils_nativeCallAndroidToast
  (JNIEnv * env, jobject obj)

    jclass j_class = env->FindClass("com/r/ndk/NativeUtils");
    jobject j_object = env->AllocObject(j_class);

    jmethodID j_method_id1 = env->GetMethodID(j_class, "showToast","(Ljava/lang/String;)V");//获取方法ID
    jmethodID j_method_id2 = env->GetStaticMethodID(j_class, "getStringFromJavaWithStatic","(Ljava/lang/String;)Ljava/lang/String;");//获取静态方法ID
    jstring j_string1=(jstring)env->NewStringUTF(" c++ string ");
    jstring j_string2 = (jstring)env->CallStaticObjectMethod(j_class, j_method_id2 ,j_string1);//调用静态方法
    env->CallVoidMethod(j_object,j_method_id1,j_string2);//调用方法

  

Java代码

    // 提示信息,提供JNI调用
    public void showToast(String msg) 
        Toast.makeText(mContext, msg, Toast.LENGTH_SHORT).show();
    

Java调用代码

    //模拟native层调用安卓Toast
    mNativeUtils.nativeCallAndroidToast();

效果:

嗯,到这里基本上Java和C++之间的相互都过了一遍,对于拓展方面可以参考JNI中文文档进行学习,开发中多少都会遇到坑,多查查资料,解决之后共享出来,一起进步。

源码下载

以上是关于Android JNI/NDK开发JNI实现C/C++与Android/JAVA相互调用的主要内容,如果未能解决你的问题,请参考以下文章

AS2.2使用CMake方式进行JNI/NDK开发

JNI/NDK开发指南—— JNI开发流程及HelloWorld

JNI/NDK开发指南——JNI调用性能測试及优化

JNI/NDK开发指南——JNI调用性能测试及优化

JNI/NDK开发指南——JNI调用性能测试及优化

Android jni/ndk编程二:jni数据类型转换(primitive,String,array)