JAVA有個垃圾回收機制,沒有引用的對象會被虛擬機在整理內存時進行回收,所以使用弱引用java對象時,而實際調用為持久對象時將會出現問題。並且這個問題有時會隨機出現,這個會內存整理回收時機有關。
下面,說下對於在JNI中使用JAVA中對象代碼編寫方法:
JAVA類方法編寫:
- public class TestModule{
- static native boolean native_init(WeakReference<TestModule> wo);
-
- /* 構造函數中傳遞JAVA弱應用對象給JNI層 */
- public TestModule() {
- native_init(new WeakReference<TestModule>(this));
- }
-
- ...
-
- /* 提供給JNI調用的JAVA方法 */
- static String native_callback(Object wo, int i, String json) {
- try {
- WeakReference<CDCAModule> r = (WeakReference<TestModule>) wo;
- CDCAModule m = r.get();
- if (m != null)
- return m.callback.callback(i, json);
- } catch (Throwable e) {
- e.printStackTrace();
- }
- return null;
- }
- }
JNI 編寫方法:
- static jobject g_java_peer = NULL;
- static jmethodID g_callback = NULL;
-
- /* 對應JAVA層native_init函數 */
- JNIEXPORT jboolean JNICALL Java_com_test_TestModule_native_init(JNIEnv *e,
- jclass clazz, jobject wo) {
- if (wo == NULL)
- throw_runtime_exception(e, "");
-
- /**
- * 如果這個JAVA對象,垃圾回收時不予釋放掉,使用弱引用轉換成強引用對象,如此
- * 則僅用戶進行主動釋放時此對象才無效
- */
- if (g_java_peer == NULL)
- g_java_peer = e->NewGlobalRef(wo);
-
- /* 獲取得到JAVA中定義的方法 */
- if ((g_callback = e->GetStaticMethodID(clazz, "native_callback",
- "Ljava/lang/Object;ILjava/lang/String")) == NULL)
- return JNI_FALSE;
- return JNI_TRUE;
- }
-
- char*jniCallJavaTestModuleMethod(int id, const char*json, char*buf, int len) {
- JNIEnv*e = NULL;
- jstring ret = NULL;
- char *p = NULL;
- int rlen;
-
- if (id < 0 || buf == NULL || len <= 0)
- return NULL;
- /* 這裡注意,JNIEnv是一個線程相關的變量,所以使用javaAttachThread保證取得當前線程的Jni環境變量*/
- if ((e = javaAttachThread("test-thread")) == NULL)
- return NULL;
-
- /* 調用JAVA方法 */
- if ((ret = (jstring) e->CallStaticObjectMethod((jclass)g_java_peer, g_callback,
- json ? e->NewStringUTF(json) : NULL)) == NULL)
- return NULL;
- p = buf;
- if ((rlen = e->GetStringLength(ret)) > len) {
- char*p = (char*)calloc(1,rlen);
- if (p == NULL)
- return NULL;
- }
- e->GetStringUTFRegion(ret, 0, rlen, p);
- return p;
- }
在JNI中提供三種類型的引用:
1、Local Reference 本地引用,函數調用時傳入jobject或者jni函數創建的jobejct,其特點就是一旦JNI層函數返回,jobject就被垃圾回收掉,所以需要注意其生命周期。可以強制調用DeleteLocalRef進行立即回收。
jstring pathStr = env->NewStringUTF(path)
....
env->DeleteLocalRef(pathStr);
2、Global Reference 全局引用 ,這種對象如不主動釋放,它永遠都不會被垃圾回收
創建: env->NewGlobalRef(obj);
釋放: env->DeleteGlobalRef(obj)
3、Weak Global Reference 弱全局引用,一種特殊的 Global Reference ,在運行過程中可能被垃圾回收掉,所以使用時請務必注意其生命周期及隨時可能被垃圾回收掉,比如內存不足時。
使用前可以利用JNIEnv的 IsSameObject 進行判定它是否被回收
env->IsSameObject(obj1,obj2);
更多Android相關信息見Android 專題頁面 http://www.linuxidc.com/topicnews.aspx?tid=11