欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Android JNI 調(diào)用時(shí)緩存字段和方法ID示例

 更新時(shí)間:2018年07月10日 14:56:37   作者:Glumes Blog  
這篇文章主要介紹了Android JNI 調(diào)用時(shí)緩存字段和方法ID示例,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧

在 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ì)緩存失效。

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)行緩存。

比如:

public class CacheFieldAndMethodOps extends BaseOperation {
 
 static {
  initCacheMethodId(); // 靜態(tài)代碼塊中進(jìn)行緩存
 }
 private static native void initCacheMethodId();
}

在靜態(tài)代碼塊中,可以將所需要的字段 id 或者方法 id 緩存成全局變量。

具體代碼如下:

// 全局變量,作為緩存方法 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ì)多次查找的情況。

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í)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

最新評(píng)論