在 JNI 去調(diào)用 Java 的方法和訪問(wèn)字段時(shí),最先要做的操作就是獲得對(duì)應(yīng)的類以及對(duì)應(yīng)的方法 id。
事實(shí)上,通過(guò) FindClass 、GetFieldID、GetMethodID 去找到對(duì)應(yīng)的信息是很耗時(shí)的,如果方法被頻繁調(diào)用,那么肯定不能每次都去查找對(duì)應(yīng)的信息,有必要將它們緩存起來(lái),在下一次調(diào)用時(shí),直接使用緩存內(nèi)容就好了。
緩存有兩種方式,分別是使用時(shí)緩存和初始化時(shí)緩存。
使用時(shí)緩存
使用時(shí)緩存,就是在調(diào)用時(shí)查找一次,然后將它緩存成 static 變量,這樣下次調(diào)用時(shí)就已經(jīng)被初始化過(guò)了。
直到內(nèi)存釋放了,才會(huì)緩存失效。
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 變量進(jìn)行緩存 // 兩種方法都行 // 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); } |
通過(guò)聲明為 static 變量進(jìn)行緩存。但這種緩存方式顯然有弊端,當(dāng)多個(gè)調(diào)用者同時(shí)調(diào)用時(shí),就會(huì)出現(xiàn)緩存多次的情況,并且每次調(diào)用時(shí)都要檢查是否緩存過(guò)了。
初始化時(shí)緩存
在初始化時(shí)緩存,就是在類加載時(shí),進(jìn)行緩存。當(dāng)類被加載進(jìn)內(nèi)存時(shí),會(huì)先調(diào)用類的靜態(tài)代碼塊,所以可以在類的靜態(tài)代碼塊中進(jìn)行緩存。
比如:
1
2
3
4
5
6
7
|
public class CacheFieldAndMethodOps extends BaseOperation { static { initCacheMethodId(); // 靜態(tài)代碼塊中進(jìn)行緩存 } private static native void initCacheMethodId(); } |
在靜態(tài)代碼塊中,可以將所需要的字段 id 或者方法 id 緩存成全局變量。
具體代碼如下:
1
2
3
4
5
6
7
8
9
10
|
// 全局變量,作為緩存方法 id jmethodID InstanceMethodCache; // 初始化加載時(shí)緩存方法 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 緩存成全局變量了,這樣再調(diào)用時(shí),就不要再進(jìn)行一次查找了,并且避免了多個(gè)線程同時(shí)調(diào)用會(huì)多次查找的情況。
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); } |
小結(jié)
可以看出,如果不能預(yù)先知道方法和字段所在類的源碼,那么在使用時(shí)緩存比較合理。但如果知道的話,在初始化時(shí)緩存優(yōu)點(diǎn)較多,既避免了每次使用時(shí)檢查,還避免了在多線程被調(diào)用的情況。
具體示例代碼可參考我的 Github 項(xiàng)目 https://github.com/glumes/AndroidDevWithCpp,歡迎 Star。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持服務(wù)器之家。
原文鏈接:https://glumes.com/post/android/android-jni-cache-fieldid-and-methodid/