我爱编程

C_Cpp-C-JNI-NDK-Java

2017-08-27  本文已影响17人  lioilwin

一.介绍

JNI(Java Native Interface)是Java提供的编程接口,
通过JNIEnv指针,C/C++和java代码可以相互调用传递数据!

NDK是Android开发C/C++工具集(代码库和编译工具),能将C/C++编程成不同平台(CUP/ABI)的so库

jni.h文件位置:
1.Android-ndk/platforms/android-版本/arch-arm/usr/include/jni.h
2.JDK/include/jni.h

二.C/C++和Java数据类型映射

在jni.h中 
typedef为C/C++类型定义别名,和java类型建立映射

1.基本数据类型映射
typedef unsigned char   jboolean;       /* unsigned 8 bits */
typedef signed char     jbyte;          /* signed 8 bits */
typedef unsigned short  jchar;          /* unsigned 16 bits */
typedef short           jshort;         /* signed 16 bits */
typedef int             jint;           /* signed 32 bits */
typedef long long       jlong;          /* signed 64 bits */
typedef float           jfloat;         /* 32-bit IEEE 754 */
typedef double          jdouble;        /* 64-bit IEEE 754 */

2.类对象映射
2.1在C++中(全都继承于_jobject类)
class _jobject {};
class _jclass : public _jobject {};
class _jstring : public _jobject {};
class _jarray : public _jobject {};
class _jobjectArray : public _jarray {};
class _jbooleanArray : public _jarray {};
class _jbyteArray : public _jarray {};
class _jcharArray : public _jarray {};
class _jshortArray : public _jarray {};
class _jintArray : public _jarray {};
class _jlongArray : public _jarray {};
class _jfloatArray : public _jarray {};
class _jdoubleArray : public _jarray {};
class _jthrowable : public _jobject {};
typedef _jobject*       jobject;
typedef _jclass*        jclass;
typedef _jstring*       jstring;
typedef _jarray*        jarray;
typedef _jobjectArray*  jobjectArray;
typedef _jbooleanArray* jbooleanArray;
typedef _jbyteArray*    jbyteArray;
typedef _jcharArray*    jcharArray;
typedef _jshortArray*   jshortArray;
typedef _jintArray*     jintArray;
typedef _jlongArray*    jlongArray;
typedef _jfloatArray*   jfloatArray;
typedef _jdoubleArray*  jdoubleArray;
typedef _jthrowable*    jthrowable;
typedef _jobject*       jweak;

2.2在C中(没有类概念,全都是void*别名)
typedef void*           jobject;
typedef jobject         jclass;
typedef jobject         jstring;
typedef jobject         jarray;
typedef jarray          jobjectArray;
typedef jarray          jbooleanArray;
typedef jarray          jbyteArray;
typedef jarray          jcharArray;
typedef jarray          jshortArray;
typedef jarray          jintArray;
typedef jarray          jlongArray;
typedef jarray          jfloatArray;
typedef jarray          jdoubleArray;
typedef jobject         jthrowable;
typedef jobject         jweak;

三.在java中调用C/C++函数

1.在java中定义native方法(C/C++函数接口)


package com.example;
public class JavaDemo { 
    public native String getHello();
}

2.实现C/C++函数接口


#include <stdio.h>
#include <jni.h>

// JNIEnv*封装很多java方法,jobject obj代表调用该函数的java对象
jstring Java_com_example_JavaDemo_getHello(JNIEnv* env,jobject obj){
    return (*env)->NewStringUTF(env,"hello...");
}

3.使用NDK编译C/C++,生成libhello.so

在Android Studio编译C/C++
https://developer.android.google.cn/studio/projects/add-native-code.html
Google官方文档中国版, 专为中国开发者建的网站!

4.加载so库,调用C/C++函数


public class Main{
    
    // 加载C/C++库libhello.so, 去除前缀lib
    static{
        System.loadLibrary("hello");
    }
    
    public void onCreate() {        
        new JavaDemo().getHello(); // 调用C/C++函数
    }
}

四.在C/C++中调用java方法


// 调用java方法, 把Java字符串解析为C语言字符串
char* Jstring2CStr(JNIEnv* env, jstring jstr){
    // 1.获取String类(java反射)
    jclass clsstring = (*env)->FindClass(env,"java/lang/String");
    
    // 2.获取String.getBytes方法(java反射)
    jmethodID mid = (*env)->GetMethodID(env,clsstring, "getBytes", "(Ljava/lang/String;)[B");
    
    // 3.调用getBytes方法(java反射),把java字符串变为字节数组jbyteArray
    jstring strencode = (*env)->NewStringUTF(env,"GB2312");
    jbyteArray barr= (jbyteArray)(*env)->CallObjectMethod(env,jstr,mid,strencode);
    
    // 4.让jbyte*(char*别名) ba引用Java数组jbyteArray barr
    jsize alen = (*env)->GetArrayLength(env,barr);
    jbyte* ba = (*env)->GetByteArrayElements(env,barr,JNI_FALSE);
    
    // 5.把Java端数组复制到本地内存(char*指向的)
    char* rtn = NULL;
    if(alen > 0){
        rtn = (char*)malloc(alen+1);
        memcpy(rtn,ba,alen);
        rtn[alen]=0;
    }
    
    // 6.释放jbyte* ba对数组jbyteArray barr的引用, 好让java回收jbyteArray barr占用的内存
    (*env)->ReleaseByteArrayElements(env,barr,ba,0);
    return rtn ;
}

简书: http://www.jianshu.com/p/52b3774fbe2b
CSDN博客: http://blog.csdn.net/qq_32115439/article/details/72567183
GitHub博客:http://lioil.win/2017/05/19/C_JNI_NDK_Java.html
Coding博客:http://c.lioil.win/2017/05/19/C_JNI_NDK_Java.html

上一篇下一篇

猜你喜欢

热点阅读