这两个函数的区别在于GetStaticMethodID是用来获取静态方法的定义,GetMethodID则是获取非静态的方法定义。这两个函数都需要提供四个参数:env就是初始化虚拟机得到的JNI环境;第二个参数class是对象的类定义,也就是第一步得到的obj;第三个参数是方法名称;最重要的是第四个参数,这个参数是方法的定义。
因为我们知道Java中允许方法的多态,仅仅是通过方法名并没有办法定位到一个具体的方法,因此需要第四个参数来指定方法的具体定义。但是怎么利用一个字符串来表示方法的具体定义呢?JDK中已经准备好一个反编译工具javap,通过这个工具就可以得到类中每个属性、方法的定义。下面就来看看jni.test.Demo的定义:
打开命令行窗口并运行 javap -s -p jni.test.Demo 得到运行结果如下:
Compiled from Demo.java
public class jni.test.Demo
extends java.lang.Object
{
public static int COUNT;
/* I */
public java.lang.String msg;
/* Ljava/lang/String;
*/
private int counts[];
/* [I */
public jni.test.Demo();
/* ()V */
public jni.test.Demo
(java.lang.String);
/* (Ljava/lang/String;)V */
public java.lang.String getMessage();
/* ()Ljava/lang/String; */
public int getCounts()[];
/* ()[I */
public void setCounts(int[]);
/* ([I)V */
public void throwExcp() throws
java.lang.IllegalAccessException;
/* ()V */
static {};
/* ()V */
} |
我们看到类中每个属性和方法下面都有一段注释。注释中不包含空格的内容就是第四个参数要填的内容(关于javap具体参数请查询JDK的使用帮助)。下面这段代码演示如何访问jni.test.Demo的getMessage方法:
/*
假设我们已经有一个
jni.test.Demo的实例obj
*/
jmethodID mid;
jclass cls =
(*env)-> GetObjectClass (env, obj);
//获取实例的类定义
mid=(*env)->GetMethodID
(env,cls,"getMessage","
()Ljava/lang/String;
");
/*如果mid为0表示获取方法定义失败*/
jstring msg = (*env)->
CallObjectMethod(env, obj, mid);
/*
如果该方法是静态的方法那只需要将
最后一句代码改为以下写法即可:
jstring msg = (*env)->
CallStaticObjectMethod
(env, cls, mid);
*/ |
3.调用方法
为了调用对象的某个方法,可以使用函数CallMethod或者CallStaticMethod(访问类的静态方法),根据不同的返回类型而定。这些方法都是使用可变参数的定义,如果访问某个方法需要参数时,只需要把所有参数按照顺序填写到方法中就可以。在讲到构造函数的访问时,将演示如何访问带参数的构造函数。
访问类属性
访问类的属性与访问类的方法大体上是一致的,只不过是把方法变成属性而已。
1.获取指定对象的类(jclass)
这一步与访问类方法的第一步完全相同,具体使用参看访问类方法的第一步。
2.读取类属性的定义(jfieldID)
在JNI中是这样定义获取类属性的方法的:
jfieldID
(JNICALL *GetFieldID)
(JNIEnv *env, jclass clazz,
const char *name, const char *sig);
jfieldID (JNICALL *GetStaticFieldID)
(JNIEnv *env, jclass clazz,
const char *name, const char *sig); |
这两个函数中第一个参数为JNI环境;clazz为类的定义;name为属性名称;第四个参数同样是为了表达属性的类型。前面我们使用javap工具获取类的详细定义的时候有这样两行:
public java.lang.String msg;
/* Ljava/lang/String; */ |
其中第二行注释的内容就是第四个参数要填的信息,这跟访问类方法时是相同的。 |