Android开发之JNI中Java与C/C++相互调用

1.Java调用C/C++

  • 加载so。加载动态库这段代码一般是在加载类时自动加载的,通过静态代码的方式:


    loadlib.png
  • 调用本地方法。一旦确定哪些功能哪些函数需要在用C/C++语言实现,然后将这些函数生命为native类型,Java调用到JNI层中的C函数就是通过native函数映射的。类似这样:

native_method.png
  • native方法调用通过JNI会映射到C的代码,了解这个映射方法,就理解了如何从native方法调用到C代码:
    映射关系是这样的:
    native函数名 <----> 包名类名函数名
    怎么理解呢:其实就是一个寻址的过程,包名类名函数名这三级寻址过程其实就是一个映射过程,寻找一个函数名,首先通过包名确定在项目的哪个目录下,然后再通过类名确定在那个类中,再通过函数名就能定位到此函数,然后通过签名描述的映射关系,函数中的所有参数与native函数参数有一个签名描述的映射关系,查看native函数与JNI函数的签名映射关系,可以使用javap命令,如果想直接生成native函数在JNI中对应的带有签名参数的函数,可以使用javah命令。

    JNI函数签名类似这样:

native_c_method.png

2.C/C++ 调用Java

举一个从JNI C调用系统提供的MD5签名Java方法的例子:

//获取签名文件数据流
jniClass = (*env)->GetObjectClass(env, obj_package_info);
jfieldID fieldID_signatures = (*env)->GetFieldID(env, jniClass, "signatures", "[Landroid/content/pm/Signature;");
if (fieldID_signatures == NULL) {
    LOGD("get signatures field failed");
    exitApp(env);
    return;
}
jobjectArray signatures = (*env)->GetObjectField(env, obj_package_info, fieldID_signatures);
jobject signature = (*env)->GetObjectArrayElement(env, signatures, 0);
if (signature == NULL) {
    LOGD("get signature data failed");
    exitApp(env);
    return;
}
//将签名数据转化为字节数组
jniClass = (*env)->GetObjectClass(env, signature);
jniMethod = (*env)->GetMethodID(env, jniClass, "toByteArray", "()[B");
jobject obj_sign_byte_array = (*env)->CallObjectMethod(env, signature, jniMethod);

jniClass = (*env)->FindClass(env, "java/security/MessageDigest");
if (jniClass == NULL) {
    LOGD("get MessageDigest class failed");
    exitApp(env);
    return ;
}
jniMethod = (*env)->GetStaticMethodID(env, jniClass, "getInstance", "(Ljava/lang/String;)Ljava/security/MessageDigest;");
if (jniMethod == NULL) {
    LOGD(" get MessageDigest.getInstance() method failed");
    exitApp(env);
    return;
}

//调用getInstance()方法
jobject md5obj = (*env)->CallStaticObjectMethod(env, jniClass, jniMethod, (*env)->NewStringUTF(env, "md5"));

jniMethod = (*env)->GetMethodID(env, jniClass, "update", "([B)V");
if (jniMethod == NULL) {
    LOGD("get MD5 update() method failed");
    exitApp(env);
    return;
}

//调用update()方法
(*env)->CallVoidMethod(env, md5obj, jniMethod, obj_sign_byte_array);

jniMethod = (*env)->GetMethodID(env, jniClass, "digest", "()[B");
if (jniMethod == NULL) {
    LOGD("get MD5 digest() method failed");
    exitApp(env);
    return;
}
//调用digest()方法
jbyteArray obj_array_sign = (jbyteArray)(*env)->CallObjectMethod(env, md5obj, jniMethod);
  1. 根据方法签名获取Java的方法,

    • 根据对象实例调用GetObjectClass拿到需要调用函数做属的类,比如

        jniClass = (*env)->GetObjectClass(env, signature);此处signature为类的对象实例
      
    • 根据函数所属的类调用GetMethodID方法拿到函数对象,此处需要传入几个参数,分别是:函数所属的类,函数名,函数参数对应的签名描述,函数返回类型对应的签名描述

        jniMethod = (*env)->GetMethodID(env, jniClass, "toByteArray", "()[B");
      
  2. 调用Java的方法。调用CallObjectMethod方法调用Java方法,此处需要传入函数对象及函数调用所需要的实参,如不需要实参,可以不写。

     (*env)->CallObjectMethod(env, signature, jniMethod);
    

至此,就完成了C调用Java函数的过程。

3.几个特别有用的工具

  • JNI生成头文件定义
    使用javah命令执行:

    • 首先在CMD终端或Shell终端进入到JNI Java文件的包名所在目录下
      比如类似这样的目录:src/main/java/包名/TestJni

    • 根据TestJni.java中的native方法生成对应的JNI C包含函数声明的头文件

    • 进入到src/main/java目录下,然后执行 javah -jni 包名.TestJni 这样会在 与TestJni所在目录下生成.h文件。

  • 生成类中函数的签名描述

    使用javap查看某个Java类中函数对应JNI的签名描述。比如查看java.lang.String中函数的内部签名:
    执行命令javap -s java.lang.String 查看所有函数的签名描述:

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

推荐阅读更多精彩内容