首页 > Coding > 多语言编程之 C and Java:JNI进阶–字段访问与方法回调

多语言编程之 C and Java:JNI进阶–字段访问与方法回调

2016年7月9日 发表评论 阅读评论

在上一章(《多语言编程之 C and Java:JNI 30 分钟入门》)中讲解了 JNI 的基本使用。这一章将讨论一些更加实用的用法:如何在 JNI 中访问 Java 对象的成员变量或静态变量,以及回调 Java 中的成员方法或静态方法。

1 JNI 访问 Java 成员变量

1.1 实例

Java 实现

这个 Java 类中包含 3 个私有成员变量,名为 age 的 int 型变量, 名为 name 的 String 型变量和一个名为 years 的 int 型静态变量。我们将在使用 C 来访问或修改这些成员变量。

C 实现

1.2 访问 Java 变量

在 JNI 中我们获取了这 3 个变量,并对 age 变量进行了修改。
由这段代码可以看出,访问 Java 成员变量的一般步骤为:

  1. 通过 GetObjectClass() 获取 Java 类的引用。
  2. 通过 GetFieldID()/GetStaticFieldID() 从 Java 类中获取 FieldID 。
  3. 基于 FieldID, 通过 Get<type>Field()/GetStatic<type>Field() 取回变量。
  4. 基于 FieldID, 通过 Set<type>Field()/GetStatic<type>Field() 修改变量。

GetFieldID()/GetStaticFieldID() 原型如下:

此函数可以通过字段名和字段标识(signature)获取类或对象的字段 ID.

Table 1: Java 字段类型标识
Java 类型 signature 例子
Boolean Z
Byte B
Char C
Int I
Short S
Long J
Float F
Double D
Void V
Object[注] L<package>/<class>; 如 String 为 Ljava/lang/String;
Array[注] [<type> 如 int[] 为 [I

:数组和引用类型的对象(Object),都是以分号(;)结尾。对于嵌套类,使用 $ 符号来表示。

Get<type>Field()/Set<type>Field() 有一组函数,用于从对象中获取字段值或给变量设置值。

<type> 为 Java 原始类型名称。
还有一组函数用于获取或设置静态字段 : GetStatic<type>Field() / SetStatic<type>Field() 。

2 JNI 访问 Java 成员方法

我们可以在 native code 中访问 Java 的成员方法和静态成员方法。这种访问也可以称之为回调。
和访问 Java 成员变量类似,访问Java 成员方法的一般步骤为:

  1. 通过 GetObjectClass() 获取 Java 对象的类。
  2. 通过 GetMethodID() 获取 MethodID.
  3. 基于 MethodID, 调用 Call<type>Method() / CallVoidMethod() / CallObjectMethod(), 这些方法会返回 <type> 类型的值。 相应地,有 CallStatic<type>Method() / CallStaticVoidMethod() / CallStaticObjectMethod()用于回调静态方法。

GetMethodID() 原型如下:

最后一个参数 sig 为方法的标识(signature)。对于这个标识,我们并不陌生,在上一章,我们使用 javah 命令生成的头文件中,对方法有一个名为 Signature 的注释,就是这个标识。 方法标识以括号 "(" 开始,一对括号内部为方法参数的标识,括号后面跟返回值的标识。如方法

的标识为: (II)V 。
除了参考 .h 文件的注释,还可以使用 Java 工具 javap 来查看方法标识。

在得到 MethodID 后,即可以通过如下函数来调用 Java 成员方法或表态方法:

3 JNI 创建 Java 对象

通过JNI提供的 NewObject() / NewObjectArray() 函数,可以在 JNI 中创建 Java 对象(jobject),并传递回 Java 程序。

3.1 在 JNI 中调用 Java 对象构造器(Constructor)

这个过程和上一节调用 Java 方法没有什么不同,只不过,GetMethodID() 中,方法名和标识不一样,这里分别为 "<init>" 和 "V" .获取 MethodID 后,就可以通过 NewObject() 来创建 Java 对象。
创建 Java 对象相关的函数还有:

此外,还有如下函数可以创建 Java 数组对象:

3.2 示例

定义 Person 类,有构造函数 Person(String, int):

并有 native 方法,将 name 和 age 传入 JNI 中,用于实例化一个 Person 对象并传回 Java:

则在 native 实现中,有如下过程:

4 附录

4.1 本章源码

TransVar.java

 

Person.java

transvar_TransVar.c

4.2 参考

  1. 本文目前尚无任何评论.
  1. 本文目前尚无任何 trackbacks 和 pingbacks.