IntelliJ IDEA平台下JNI编程(四)—本地C代码访问JAVA对象

转载请注明出处:【huachao1001的简书:http://www.jianshu.com/users/0a7e42698e4b/latest_articles】

本文主要针对C代码中访问JVM中对象的普通变量、静态属性、普通函数、静态函数进行举例讲解,通过本文的学习将进一步理解JNIEnv在本地代码和Java之间的重要性。有了前面几篇文章的基础,学习起本文来将更容易。好了,接下来往下学习吧~。

1. 访问实例对象的属性

下面实现一个简单的功能:在本地C代码中,修改Java类实例的属性值。首先在Java类中定义2个简单的属性,代码如下:

package com.huachao.java;

/**
 * Created by HuaChao on 2017/03/22.
 */
public class HelloJNI { 
    private int number = 88;
    private String message = "Hello from Java";


    static {
        System.loadLibrary("HelloJNI");
    }

    private native void modifyField();

    public static void main(String[] args) {
        HelloJNI obj = new HelloJNI();
        obj.modifyField();
        System.out.println("Java类中,number的值为:" + obj.number);
        System.out.println("Java类中,message属性为:" + obj.message);
    }

}

上面代码很简单,就是定义了一个本地函数modifyField(),还有两个属性number和message。接下来,生成c代码的头文件后,c代码对应的函数代码如下:

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

JNIEXPORT void JNICALL Java_com_huachao_java_HelloJNI_modifyField
  (JNIEnv * env,jobject thisObj){
       //获取实例对象类的引用
       jclass thisClass = (*env)->GetObjectClass(env, thisObj);

       // 获取实例对象的nunber对应的属性ID,int类型
       jfieldID fidNumber = (*env)->GetFieldID(env, thisClass, "number", "I");
       if (NULL == fidNumber) return;

       //获取实例对象中对应的属性的值
       jint number = (*env)->GetIntField(env, thisObj, fidNumber);
       printf("在C代码中,number属性值为:%d\n", number);

       // 修改number的值
       number = 99;
       (*env)->SetIntField(env, thisObj, fidNumber, number);

       // 同理,获取类对象中的message的属性ID
       jfieldID fidMessage = (*env)->GetFieldID(env, thisClass, "message", "Ljava/lang/String;");
       if (NULL == fidMessage) return;

       // 获取给定属性ID的属性对象
       jstring message = (*env)->GetObjectField(env, thisObj, fidMessage);

       // 创建C语言中的字符串
       const char *cStr = (*env)->GetStringUTFChars(env, message, NULL);
       if (NULL == cStr) return;

       printf("在C代码中, 字符串message为:%s\n", cStr);
       (*env)->ReleaseStringUTFChars(env, message, cStr);

       // 创建新的C字符串,并赋值给实例对象的属性
       message = (*env)->NewStringUTF(env, "Hello from C");
       if (NULL == message) return;

       // 修改实例对象的属性值
       (*env)->SetObjectField(env, thisObj, fidMessage, message);
 }

编译为dll文件后,运行结果如下:

Java类中,number的值为:99
Java类中,message属性为:Hello from C
在C代码中,number属性值为:88
在C代码中, 字符串message为:Hello from Java

看到运行结果觉得挺奇怪,因为先调用modifyField函数,在打印结果中先打印了Java中的代码后打印C中的代码。现在我也还不清楚具体原因,但从结果上看,是先执行本地代码的。因此打印的先后顺序暂时可不管他,我目前认为是c的输出流和Java的输出流在输出到IntelliJ的控制台时合并的先后顺序造成的吧,如果有小伙伴知道原因,可以在评论中指出。

下面对代码进行解释,访问实例对象的属性可分为如下步骤:

  1. 首先是通过jobject指代的实例对象获取实例对象在Java虚拟机中对应的Class对象,即对应于jclass类型。这一步通过调用JNIEnv对象的jclass GetObjectClass(JNIEnv *env, jobject obj);函数来完成。
  2. 接下来,获取用于标识属性的ID对象,即jfieldID类型对象。通过调用JNIEnvjfieldID GetFieldID(JNIEnv *env, jclass cls, const char *name, const char *sig);函数实现。GetFieldID函数前两个参数不用解释,第三个参数是表示Java类中属性的名称,第四个参数表示属性的描述符。关于描述符在《IntelliJ IDEA平台下JNI编程(二)—类型映射》已经做过介绍,不清楚的可以回去查阅。
  3. 接下来是根据属性的ID获取属性的值,通过调用JNIEnvNativeType Get<type>Field(JNIEnv *env, jobject obj, jfieldID fieldID);来实现。此时可获取Java实例对象的的属性。
  4. 如果需要对实例对象的属性进行修改,调用JNIEnvvoid Set<type>Field(JNIEnv *env, jobject obj, jfieldID fieldID, NativeType value);实现。

可以看到,在C代码中访问Java实例对象的属性与Java中通过反射的方法访问非常类似。只是这里用到了ID的方式来进行访问,而不是Java反射中的Field对象来访问。这主要是在跨越两种语言相互访问内存时,通过唯一标识码来间接访问效率更高也更简单。

2. 访问类的静态变量

因为静态变量是属于类对象,即Class对象。因此在访问静态变量时,c代码获取到对应的jclass对象后,即可访问。下面通过一个简单例子说明,首先在HelloJNI.java类中添加一个静态变量。

 private static double price = 55.66;

在c代码的Java_com_huachao_java_HelloJNI_modifyField中,代码如下:

JNIEXPORT void JNICALL Java_com_huachao_java_HelloJNI_modifyField
  (JNIEnv * env,jobject thisObj){

     // 获取实例对象类的引用
       jclass thisClass = (*env)->GetObjectClass(env, thisObj);

      // 获取静态变量,并修改静态变量的值
      jfieldID fidNumber = (*env)->GetStaticFieldID(env, thisClass, "price", "D");
      if (NULL == fidNumber) return;
      jdouble number = (*env)->GetStaticDoubleField(env, thisClass, fidNumber);
      printf("In C, the double is %f\n", number);
      number = 77.88;
      (*env)->SetStaticDoubleField(env, thisClass, fidNumber, number);
 }

运行结果如下:

Java类中,price的值为:77.88
In C, the double is 55.660000

静态变量的访问过程跟实例变量类似,只是对应的函数多了static部分,这里不再继续解释。

3. 调用Java的对象方法和静态方法

调用Java的函数过程与访问Java属性过程一致,有了前面的经验后,学习起调用函数来更容易了。下面同样通过一个简单例子来学习。首先在Java类中,定义3个具有代表性的函数:有返回值的对象方法、无返回值的对象方法,以及有返回值的静态方法。

package com.huachao.java;

/**
 * Created by HuaChao on 2017/03/22.
 */
public class HelloJNI {


    static {
        System.loadLibrary("HelloJNI");
    }

    private native void nativeMethod();

    private int sum(int n1, int n2) {
        return n1 + n2;
    }

    private static double avg(int n1, int n2) {
        return (double) ((n1 + n2) * 1.0f / 2);
    }

    private void display() {
        System.out.println("invoke display()");

    }

    public static void main(String[] args) {
        HelloJNI obj = new HelloJNI();
        obj.nativeMethod();
    }

}

在c代码中,同样是获取方法的id后,再调用方法。

JNIEXPORT void JNICALL Java_com_huachao_java_HelloJNI_nativeMethod
  (JNIEnv * env,jobject thisObj){

     // 获取实例对象类的引用
     jclass thisClass = (*env)->GetObjectClass(env, thisObj);
     //获取方法ID,注意普通函数和静态函数的获取区别
     jmethodID sumId = (*env)->GetMethodID(env, thisClass, "sum", "(II)I");
     jmethodID avgId = (*env)->GetStaticMethodID(env, thisClass, "avg", "(II)D");
     jmethodID displayId = (*env)->GetMethodID(env, thisClass, "display", "()V");

     if (NULL == sumId) return;
     if (NULL == avgId) return;
     if (NULL == displayId) return;
     
    //调用方法,注意调用静态方程和普通方法的区别
     (*env)->CallVoidMethod(env, thisObj, displayId);
     double avgRs=(*env)->CallStaticDoubleMethod(env,thisClass,avgId,1,3);
     printf("avg(1,3)=%f\n",avgRs);
     int sumRs=(*env)->CallIntMethod(env,thisObj,sumId,1,3);
     printf("sum(1,3)=%d\n",sumRs);
 }

运行结果如下:

invoke display()
avg(1,3)=2.000000
sum(1,3)=4

代码中解释很清楚了,这里就不再解释。

4. 调用父类中被覆盖的方法

构造一个简单的继承关系:

package com.huachao.java;

/**
 * Created by HuaChao on 2017/03/22.
 */
class Parent {
    protected void sayHello() {
        System.out.println("Hello in Parent");
    }
}

public class HelloJNI extends Parent{


    static {
        System.loadLibrary("HelloJNI");
    }

    private void sayHello() {
        System.out.println("Hello in Child");
    }

    private native void nativeMethod();

    public static void main(String[] args) {
        HelloJNI obj = new HelloJNI();
        obj.nativeMethod();
    }

}


在c代码中,通过子类的实例调用父类的被覆盖的函数方法如下:

JNIEXPORT void JNICALL Java_com_huachao_java_HelloJNI_nativeMethod
  (JNIEnv * env,jobject thisObj){

     // 在JVM中查找指定的Class对象
     jclass parentClass=(*env)->FindClass( env,"com/huachao/java/Parent");
     jmethodID sayHelloId = (*env)->GetMethodID(env, parentClass, "sayHello", "()V");
     if (NULL == sayHelloId) return;

     (*env)->CallNonvirtualVoidMethod(env, thisObj, parentClass,sayHelloId);

 }

运行结果如下:

Hello in Parent

注意GetMethodID函数,因为传入的是com/huachao/java/Parent对应的Class对象,因此返回的ID是指com/huachao/java/Parent类中的sayHello函数。调用父类的函数通过CallNonvirtual<TYPE>Method来实现。CallNonvirtual<TYPE>Method函数有如下几种原型:

NativeType CallNonvirtual<type>Method(JNIEnv *env, jobject obj, jclass cls, jmethodID methodID, ...);
NativeType CallNonvirtual<type>MethodA(JNIEnv *env, jobject obj, jclass cls, jmethodID methodID, const jvalue *args);
NativeType CallNonvirtual<type>MethodV(JNIEnv *env, jobject obj, jclass cls, jmethodID methodID, va_list args);

学到这里会发现,c代码中要想访问JVM中对象,必须通过JNIEnv提供的函数来间接完成。这也进一步加深了对JNIEnv的了解。

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

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,295评论 18 399
  • 什么是JNI? JNI 是java本地开发接口.JNI 是一个协议,这个协议用来沟通java代码和外部的本地代码(...
    a_tomcat阅读 2,767评论 0 54
  • 据央视某记者的报道,美国国务卿蒂勒森,4月28日在联合国的一次部长级会议上,主张对朝鲜“施加经济和外交压力”,并以...
    海天大路阅读 1,324评论 7 11
  • 亲情,与生俱来;源于血脉,但又不完全源于血缘。岁月的洗礼,会显现亲情的浓淡;利欲的考验,会证明亲情的真假。在金钱面...
    浦和0917阅读 445评论 2 2
  • 写作前十篇,基本都是个人经历和感念,并借由跟最亲密的人的关系来反映我的心智蜕变。正所谓“亲密关系是最佳道场“,其实...
    大树磕宝阅读 913评论 10 16