2.4.4 通过JNIEnv操作jobject
前面提到过一个问题,即Java的引用类型除了少数几个外,最终在JNI层都会用jobject来表示对象的数据类型,那么该如何操作这个jobject呢?
从另外一个角度来解释这个问题。一个Java对象是由什么组成的?当然是它的成员变量和成员函数了。那么,操作jobject的本质就应当是操作这些对象的成员变量和成员函数。所以应先来看与成员变量及成员函数有关的内容。
1.jfieldID和jmethodID介绍
我们知道,成员变量和成员函数都是由类定义的,它们是类的属性,所以在JNI规则中,用jfieldID和jmethodID来表示Java类的成员变量和成员函数,可通过JNIEnv的下面两个函数得到:
jfieldID GetFieldID(jclass clazz,const charname,const charsig);
jmethodID GetMethodID(jclass clazz,const charname,const charsig);
其中,jclass代表Java类,name表示成员函数或成员变量的名字,sig为这个函数和变量的签名信息。如前所示,成员函数和成员变量都是类的信息,这两个函数的第一个参数都是jclass。
在MS中是怎么使用它们的呢?来看代码,如下所示:
[—>android_media_MediaScanner.cpp:MyMediaScannerClient构造函数]
MyMediaScannerClient(JNIEnv*env,jobject client)……
{
//先找到android.media.MediaScannerClient类在JNI层中对应的jclass实例。
jclass mediaScannerClientInterface=
env->FindClass("android/media/MediaScannerClient");
//取出MediaScannerClient类中函数scanFile的jMethodID。
mScanFileMethodID=env->GetMethodID(
mediaScannerClientInterface,"scanFile",
"(Ljava/lang/String;JJ)V");
//取出MediaScannerClient类中函数handleStringTag的jMethodID。
mHandleStringTagMethodID=env->GetMethodID(
mediaScannerClientInterface,"handleStringTag",
"(Ljava/lang/String;Ljava/lang/String;)V");
……
}
在上面的代码中,将scanFile和handleStringTag函数的jmethodID保存为MyMediaScannerClient的成员变量。为什么这里要把它们保存起来呢?这个问题涉及一个关于程序运行效率的知识点:
如果每次操作jobject前都去查询jmethodID或jfieldID,那么将会影响程序运行的效率,所以我们在初始化的时候可以取出这些ID并保存起来以供后续使用。
取出jmethodID后,又该怎么用它呢?
2.使用jfieldID和jmethodID
下面再看一个例子,其代码如下所示:
[—>android_media_MediaScanner.cpp:MyMediaScannerClient的scanFile]
virtual bool scanFile(const char*path,long long lastModified,
long long fileSize)
{
jstring pathStr;
if((pathStr=mEnv->NewStringUTF(path))==NULL)return false;
/*
调用JNIEnv的CallVoidMethod函数,注意CallVoidMethod的参数:
第一个是代表MediaScannerClient的jobject对象,第二个参数是函数scanFile的jmethodID,后面是Java中scanFile的参数。
*/
mEnv->CallVoidMethod(mClient,mScanFileMethodID,pathStr,
lastModified,fileSize);
mEnv->DeleteLocalRef(pathStr);
return(!mEnv->ExceptionCheck());
}
明白了,通过JNIEnv输出CallVoidMethod,再把jobject、jMethodID和对应的参数传进去,JNI层就能够调用Java对象的函数了!
实际上JNIEnv输出了一系列类似CallVoidMethod的函数,形式如下:
NativeType Call<type>Method(JNIEnv*env,jobject obj,jmethodID methodID,……)。
其中type对应Java函数的返回值类型,例如CallIntMethod、CallVoidMethod等。
上面是针对非static函数的,如果想调用Java中的static函数,则用JNIEnv输出的CallStatic<Type>Method系列函数。
现在,我们已了解了如何通过JNIEnv操作jobject的成员函数,那如何通过jfieldID操作jobject的成员变量呢?这里,直接给出整体解决方案,如下所示:
//获得fieldID后,可调用Get<type>Field系列函数获取jobject对应的成员变量的值。
NativeType Get<type>Field(JNIEnv*env,jobject obj,jfieldID fieldID)
//或者调用Set<type>Field系列函数来设置jobject对应的成员变量的值。
void Set<type>Field(JNIEnv*env,jobject obj,jfieldID fieldID,NativeType value)
//下面我们列出一些常用的Get/Set函数。
GetObjectField()SetObjectField()
GetBooleanField()SetBooleanField()
GetByteField()SetByteField()
GetCharField()SetCharField()
GetShortField()SetShortField()
GetIntField()SetIntField()
GetLongField()SetLongField()
GetFloatField()SetFloatField()
GetDoubleField()SetDoubleField()
通过本节的介绍,相信读者已经了解了jfieldID和jmethodID的作用,也知道如何通过JNIEnv的函数来操作jobject了。虽然jobject是透明的,但有了JNIEnv的帮助,还是能轻松操作jobject背后的实际对象的。