在 JNI 去調用 Java 的方法和訪問字段時,最先要做的操作就是獲得對應的類以及對應的方法 id。
事實上,通過 FindClass 、GetFieldID、GetMethodID 去找到對應的信息是很耗時的,如果方法被頻繁調用,那么肯定不能每次都去查找對應的信息,有必要將它們緩存起來,在下一次調用時,直接使用緩存內容就好了。
緩存有兩種方式,分別是使用時緩存和初始化時緩存。
使用時緩存
使用時緩存,就是在調用時查找一次,然后將它緩存成 static 變量,這樣下次調用時就已經被初始化過了。
直到內存釋放了,才會緩存失效。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
extern "C" JNIEXPORT void JNICALL Java_com_glumes_cppso_jnioperations_CacheFieldAndMethodOps_staticCacheField(JNIEnv *env, jobject instance, jobject animal) { static jfieldID fid = NULL; // 聲明為 static 變量進行緩存 // 兩種方法都行 // jclass cls = env->GetObjectClass(animal); jclass cls = env->FindClass( "com/glumes/cppso/model/Animal" ); jstring jstr; const char *c_str; // 從緩存中查找 if (fid == NULL) { fid = env->GetFieldID(cls, "name" , "Ljava/lang/String;" ); if (fid == NULL) { return ; } } else { LOGD( "field id is cached" ); } jstr = (jstring) env->GetObjectField(animal, fid); c_str = env->GetStringUTFChars(jstr, NULL); if (c_str == NULL) { return ; } env->ReleaseStringUTFChars(jstr, c_str); jstr = env->NewStringUTF( "new name" ); if (jstr == NULL) { return ; } env->SetObjectField(animal, fid, jstr); } |
通過聲明為 static 變量進行緩存。但這種緩存方式顯然有弊端,當多個調用者同時調用時,就會出現緩存多次的情況,并且每次調用時都要檢查是否緩存過了。
初始化時緩存
在初始化時緩存,就是在類加載時,進行緩存。當類被加載進內存時,會先調用類的靜態代碼塊,所以可以在類的靜態代碼塊中進行緩存。
比如:
1
2
3
4
5
6
7
|
public class CacheFieldAndMethodOps extends BaseOperation { static { initCacheMethodId(); // 靜態代碼塊中進行緩存 } private static native void initCacheMethodId(); } |
在靜態代碼塊中,可以將所需要的字段 id 或者方法 id 緩存成全局變量。
具體代碼如下:
1
2
3
4
5
6
7
8
9
10
|
// 全局變量,作為緩存方法 id jmethodID InstanceMethodCache; // 初始化加載時緩存方法 id extern "C" JNIEXPORT void JNICALL Java_com_glumes_cppso_jnioperations_CacheFieldAndMethodOps_initCacheMethodId(JNIEnv *env, jclass type) { jclass cls = env->FindClass( "com/glumes/cppso/model/Animal" ); InstanceMethodCache = env->GetMethodID(cls, "getName" , "()Ljava/lang/String;" ); } |
在 JNI 中直接將方法 id 緩存成全局變量了,這樣再調用時,就不要再進行一次查找了,并且避免了多個線程同時調用會多次查找的情況。
1
2
3
4
5
6
7
|
extern "C" JNIEXPORT void JNICALL Java_com_glumes_cppso_jnioperations_CacheFieldAndMethodOps_callCacheMethod(JNIEnv *env, jobject instance, jobject animal) { jstring name = (jstring) env->CallObjectMethod(animal, InstanceMethodCache); const char *c_name = env->GetStringUTFChars(name, NULL); LOGD( "call cache method and value is %s" , c_name); } |
小結
可以看出,如果不能預先知道方法和字段所在類的源碼,那么在使用時緩存比較合理。但如果知道的話,在初始化時緩存優點較多,既避免了每次使用時檢查,還避免了在多線程被調用的情況。
具體示例代碼可參考我的 Github 項目 https://github.com/glumes/AndroidDevWithCpp,歡迎 Star。
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持服務器之家。
原文鏈接:https://glumes.com/post/android/android-jni-cache-fieldid-and-methodid/