Android JNI 調用時緩存字段和方法ID示例
在 JNI 去調用 Java 的方法和訪問字段時,最先要做的操作就是獲得對應的類以及對應的方法 id。
事實上,通過 FindClass 、GetFieldID、GetMethodID 去找到對應的信息是很耗時的,如果方法被頻繁調用,那么肯定不能每次都去查找對應的信息,有必要將它們緩存起來,在下一次調用時,直接使用緩存內容就好了。
緩存有兩種方式,分別是使用時緩存和初始化時緩存。
使用時緩存
使用時緩存,就是在調用時查找一次,然后將它緩存成 static 變量,這樣下次調用時就已經(jīng)被初始化過了。
直到內存釋放了,才會緩存失效。
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 變量進行緩存。但這種緩存方式顯然有弊端,當多個調用者同時調用時,就會出現(xiàn)緩存多次的情況,并且每次調用時都要檢查是否緩存過了。
初始化時緩存
在初始化時緩存,就是在類加載時,進行緩存。當類被加載進內存時,會先調用類的靜態(tài)代碼塊,所以可以在類的靜態(tài)代碼塊中進行緩存。
比如:
public class CacheFieldAndMethodOps extends BaseOperation { static { initCacheMethodId(); // 靜態(tài)代碼塊中進行緩存 } private static native void initCacheMethodId(); }
在靜態(tài)代碼塊中,可以將所需要的字段 id 或者方法 id 緩存成全局變量。
具體代碼如下:
// 全局變量,作為緩存方法 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 緩存成全局變量了,這樣再調用時,就不要再進行一次查找了,并且避免了多個線程同時調用會多次查找的情況。
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); }
小結
可以看出,如果不能預先知道方法和字段所在類的源碼,那么在使用時緩存比較合理。但如果知道的話,在初始化時緩存優(yōu)點較多,既避免了每次使用時檢查,還避免了在多線程被調用的情況。
具體示例代碼可參考我的 Github 項目 https://github.com/glumes/AndroidDevWithCpp,歡迎 Star。
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
- Android使用jni調用c++/c方法詳解
- Android中的JNI數(shù)組操作教程
- 使用Android studio編寫一個小的jni程序
- Android Jni的簡單使用詳解
- Android串口開發(fā)之使用JNI實現(xiàn)ANDROID和串口通信詳解
- android串口開發(fā)入門之搭建ndk開發(fā)環(huán)境及第一個jni調用程序
- Android JNI處理圖片實現(xiàn)黑白濾鏡的方法
- Android Studio中導入JNI生成的.so庫的實現(xiàn)方法
- Android JNI c/c++調用java的實例
- 淺談Android Studio JNI生成so庫
相關文章
Android使用recyclerview打造真正的下拉刷新上拉加載效果
這篇文章先介紹如何使用這個recyclerview,WZMRecyclerview 是一個集成了 下拉刷新、上拉加載、滑到底部自動加載、添加刪除頭尾部 四個主要功能的recyclerview,需要的朋友可以參考下2016-11-11Android仿百度谷歌搜索自動提示框AutoCompleteTextView簡單應用示例
這篇文章主要介紹了Android仿百度谷歌搜索自動提示框AutoCompleteTextView簡單應用,結合實例形式分析了AutoCompleteTextView Widget使用步驟與相關操作技巧,需要的朋友可以參考下2016-10-10老生常談Android HapticFeedback(震動反饋)
下面小編就為大家?guī)硪黄仙U凙ndroid HapticFeedback(震動反饋)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-04-04android開發(fā)環(huán)境搭建詳解(eclipse + android sdk)
這篇文章主要介紹了android開發(fā)環(huán)境搭建詳解(eclipse + android sdk),需要的朋友可以參考下2014-05-05