JNIを使っていて、C/C++側からJavaのオブジェクトを任意のタイミングで使いたいことがあります。
たとえば、ネイティブ側で処理が終わったらonCompleteメソッドを呼び出すとか、javaのInputStreamをC側で保持しておき、適宜読み込むといった場合です。
JNIメソッドの引数やFindClass/GetObjectFieldで取得したjobjectを、Cのグローバル変数(やC++のメンバ変数など)に保持しておいて、後で使おうとすると
JNI ERROR (app bug): accessed stale local reference 0x2d00025 (index 9 in a table of size 8)
のようなエラーが発生することがあります。
これは、jobjectの参照するJavaのオブジェクトが、すでにGCによって回収されている場合のエラーです。
Cでjobjectの値を保持していても、GCはそれが参照されていることを知りません。
ちゃんと、参照カウントを増やしてやりましょう。
// これでobjのグローバル参照が取得できる。objの指すJavaオブジェクトは勝手に解放されない jobject globalObj = env->NewGlobalRef(obj); // 不要になったらこれを実施しないと、永久に解放されずメモリリークする env->DeleteGlobalRef(globalObj);
基本ですね。