NDK开发系列(二)——JNI c/c++调用java

上一篇简单的介绍了JNI,简单的回顾下,java要想调用c/c++代码分为概括分为三步:
1、编写native方法,在c/c++中实现对应的c函数
2、将c代码编译成动态库
3、System.loadLibrary()对动态库进行加载

这一篇重点讲c/c++怎么访问java,为了方便开发环境切换到AndroidStudio,首先需要把AS的ndk环境给配好,详细的配置这里先不讲。AS3.0对ndk的支持已经非常友好了,以前的旧版本是用的makefile来编译动态库,新版本的是Cmake编译,其中差异还是比较大的,旧版本的我以前写过一篇文章,感兴趣可以翻出来看看。
首先,as新建一个工程


image.png

把支持c++选项勾选上,这时候如果你没有配好ndk-bound,需要到工程的属性设置里面配置:


image.png

配置好以后,as会自动生成一个ndk工程,默认生成一些示例代码,非常的友好。在main/cpp目录中存放的是c/c++代码,MainActivity中自动生成了一些示例代码,可以仿照使用。
这里我们不使用生成的代码,把代码全部干掉,重新开始。
  • c准备工作:在cpp目录下面新建c代码文件native.c,在java下面新建一个java类Man.java


    image.png

    这时候native方法会爆红,这是因为as没有检测到对应的c函数的实现,用alt+enter快捷键自动生成到native.c中,不得不说as越来越强大了,然后需要在CMakeLists.txt文件中加入我们的c文件


    image.png
# native.c
JNIEXPORT jstring JNICALL
Java_com_example_xucong_jnitest_Man_accessField(JNIEnv *env, jclass type) {

    return (*env)->NewStringUTF(env, "accessField");
}

JNIEXPORT jstring JNICALL
Java_com_example_xucong_jnitest_Man_stringFromJNI(JNIEnv *env, jobject instance) {

    return (*env)->NewStringUTF(env, "stringFromJNI");
}

运行结果:


image.png

以上是简单使用。

  • c/c++访问java属性
    首先看下Man.java这个类的代码
public class Man {

    public String name = "Tom";

    public native static String accessField();

    public native String stringFromJNI();

    static {
        System.loadLibrary("native-lib");
    }
}

这里定义了一个属性、一个静态的native方法accessField(),一个非静态的native方法stringFromJNI(),和一个静态块,用来加载as为我们编译好的so动态库,注意native-lib是as编译好动态库的名称,不包括后缀,当然这个名字我们可以CMakeLists里面修改,编译好的so库在app/build/intermediates/cmake里面
native方法有静态和非静态之分,对应的c的实现函数也有所差异:

NIEXPORT jstring JNICALL
Java_com_example_xucong_jnitest_Man_accessField(JNIEnv *env, jclass type) {

    return (*env)->NewStringUTF(env, "accessField");
}

JNIEXPORT jstring JNICALL
Java_com_example_xucong_jnitest_Man_stringFromJNI(JNIEnv *env, jobject instance) {

    return (*env)->NewStringUTF(env, "stringFromJNI");
}

c对应的java native函数中至少有两个参数,JNIEnv、jclass或者jobject,JNIEnv是JNI的运行环境,在c中一个二级结构体指针,在c++中是一级指针,他们所调用函数的方式也有不同:

JNIEXPORT jstring JNICALL
Java_com_example_xucong_jnitest_Man_accessField(JNIEnv *env, jclass type) {

    return env->NewStringUTF("accessField");
//    return (*env)->NewStringUTF(env, "accessField");
}

他们所使用的方法名都是一样,只是c++中的所有函数不在需要传env的上下文了,这是因为c++中有this上下文关键字。

jobject和jclass,这是JNI的数据类型,如果java中是非静态方法,对应的jobject,如果是静态的对应的是jclass,这两个参数是java在JNI中的映射,需要通过这两个参数来访问java。其实也好理解,如果是非静态,我们调用native方法的时候需要new一个对象,和对象实例有关,静态的方法只和Class有关。
函数的返回类型是jstring 对应的java中的String,每种java的数据类型在JNI中都有与之对应

typedef uint8_t  jboolean; /* unsigned 8 bits */
typedef int8_t   jbyte;    /* signed 8 bits */
typedef uint16_t jchar;    /* unsigned 16 bits */
typedef int16_t  jshort;   /* signed 16 bits */
typedef int32_t  jint;     /* signed 32 bits */
typedef int64_t  jlong;    /* signed 64 bits */
typedef float    jfloat;   /* 32-bit IEEE 754 */
typedef double   jdouble;  /* 64-bit IEEE 754 */

/* "cardinal indices and sizes" */
typedef jint     jsize;

#ifdef __cplusplus

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;


#else /* not __cplusplus */

/*
 * Reference types, in C.
 */
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;

数据类型分为基本数据类型和引用数据类型,引用数据类型分为jstring和jobject,还有任何数组也是jobject,这些都在jni.h源码中有。

那么进入正题:在c层调来修改Man的属性name的值,这里需要把accessField修改为非静态方法,因为name属性是非静态的,只有对象实例才有属性值。

//1.访问属性
//修改属性key的字符串
JNIEXPORT jstring JNICALL
Java_com_example_xucong_jnitest_Man_accessField(JNIEnv *env, jobject jobj) {

    //得到class
    jclass jclazz = (*env)->GetObjectClass(env,jobj);
    //jfieldID
    //签名:类型的简称
    //属性,方法
    jfieldID fid = (*env)->GetFieldID(env,jclazz,"name","Ljava/lang/String;");
    //获取key属性的值
    //注意:key为基本数据类型,规则如下
    //(*env)->GetIntField(); (*env)->Get<Type>Field();  
    jstring jstr = (*env)->GetObjectField(env,jobj,fid);
    //jstring转为 C/C++字符串
    char * c_str = (*env)->GetStringUTFChars(env,jstr,NULL);

    strcat(c_str,"android");
    //拼接完成之后,从C字符串转为jstring
    jstring jstr_new = (*env)->NewStringUTF(env,c_str);
    //修改key的属性
    //注意规则:Set<Type>Field
    (*env)->SetObjectField(env,jobj,fid,jstr_new);

    return jstr_new;
}

以上的流程和java的反射的流程非常相似,拿到class对象->获取属性id->拿到属性值->修改属性,

  • GetFieldID ,最后一个参数为数据类型的签名,name是String类型,就将String签名传入,各种数据类型的签名如下:


    image.png
  • GetObjectField,获取属性值,规则为Get<Type>Field,如果java类中的属性类型为int,则为GetIntField();

  • Get<Type>Field();修改属性的值,和GetObjectField的规则一样。
    其中要注意的是jni的字符串是没有修改的api的,需要通过c字符串来修改,再改回jstring。
    java代码:

TextView tv = findViewById(R.id.sample_text);
        Man man = new Man();
        String str = "修改前:" + man.name;
        man.accessField();
        str = str + "   修改后:" + man.name;
        tv.setText(str);

运行结果:


image.png
  • 访问java静态属性
    访问java静态属性的步骤,只是api稍有调整
    在Man.java中增加一个属性,和一个方法
    public static int age = 18;
    public native String accessStaticField();

c代码:

Java_com_example_xucong_jnitest_Man_accessStaticField(JNIEnv *env, jobject jobj) {

    //获取class
    jclass jclazz = (*env)->GetObjectClass(env,jobj);
    //获取jfieldid
    jfieldID jid = (*env)->GetStaticFieldID(env,jclazz,"age","I");
    jint jage = (*env)->GetStaticIntField(env,jclazz,jid);
    jage++;
    (*env)->SetStaticIntField(env,jclazz,jid,jage);

    return (*env)->NewStringUTF(env, "修改成功");
}

java代码:

TextView tv = findViewById(R.id.sample_text);
        Man man = new Man();
        String str = "name修改前:" + man.name;
        man.accessField();
        str = str + "\nname修改后:" + man.name;

        str += "\nage修改前:" + Man.age;
        man.accessStaticField();
        str = str + "\nage修改后:" + Man.age;
        tv.setText(str);
image.png

可以看出来,步骤和前面一样,只是访问静态属性的方法都加上了static
另外就是SetStaticIntField()方法的第二个参数类型是jclass,而不是jobject,为什么呢?这个和java的类是对应的,我们访问java的静态变量的时候,变量只和Class有关,和实例对象的应用无关,而非静态成员变量和必须要通过对象的引用来访问,在JNI中也是这个理。如果accessStaticField()方法改为static,那么JNI中实现的c方法为jclass对象可以省去jclass jclazz = (*env)->GetObjectClass(env,jobj);这一步骤。

  • C/C++D调用java方法
    直接上代码:
public native int accessMethod();

public int getRandomNum(int max) {
    return new Random().nextInt(max);
}

accessMethod()方法是进入c,c/c++中再去调用getRandomNum产生随机数,返回给accessMethod()方法。
看看JNI的实现:

//访问java方法
JNIEXPORT jint JNICALL
Java_com_example_xucong_jnitest_Man_accessMethod(JNIEnv *env, jobject jobj) {
    //获取class
    jclass jclazz = (*env)->GetObjectClass(env,jobj);
    jmethodID jmid = (*env)->GetMethodID(env,jclazz,"getRandomNum","(I)I");
    jint random = (*env)->CallIntMethod(env,jobj,jmid,100);
    return random;
}

步骤套路和前面及其的相似,不同的只是方法的调用,GetMethodID获取方法id,方法第三个参数为方法名,最后一个参数是方法签名,方法签名为对应的是jobj的java类的签名。

获取签名:
获取签名是用javap命令,打开as的terminal 可以看到javap的指令集


image.png

cd 到app/build/intermediates/debug目录下,里面有编译好的class文件,执行javap -p -s com.example.xucong.jnitest.Man指令,就能够获取参数、方法的签名,前面获取成员变量的签名的时候也可以通过这种方式:


image.png

其实这些步骤也是也可以偷懒的,可以参考我AS NDK环境变量配置的文章的末尾片段。

public native int accessStaticMethod(String filepath);
    //获取uuid随机文件名
    public static String getUUID() {
        return UUID.randomUUID().toString();
    }

//访问静态方法
//借用java api 产生一个UUID字符串,作为文件的名称
JNIEXPORT jint JNICALL
Java_com_example_xucong_jnitest_Man_accessStaticMethod(JNIEnv *env, jobject jobj, jstring jstr_file_path) {

    jclass jclazz = (*env)->GetObjectClass(env,jobj);

    jmethodID jmid = (*env)->GetStaticMethodID(env,jclazz,"getUUID","()Ljava/lang/String;");

    jstring jstr_uuid = (*env)->CallStaticObjectMethod(env,jclazz,jmid);

    char *cstr_uuid = (*env)->GetStringUTFChars(env,jstr_uuid,JNI_FALSE);
    char *cstr_file_path = (*env)->GetStringUTFChars(env,jstr_file_path,JNI_FALSE);

    char filename[100];

    sprintf(filename,cstr_file_path,cstr_uuid);

    FILE *fp = fopen(filename,"w");
    fputs(filename,fp);
    fclose(fp);

}

java :
 String path = Environment.getExternalStorageDirectory().getAbsolutePath() + File.separator + "%s.txt";
        man.accessStaticMethod(path);
image.png

注意:6.0需要动态权限

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,772评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,458评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,610评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,640评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,657评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,590评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,962评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,631评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,870评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,611评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,704评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,386评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,969评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,944评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,179评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,742评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,440评论 2 342

推荐阅读更多精彩内容