Android NDK 线程回调Java层方法

项目上刚好要在Android NDK层使用线程回调Java层方法,仅以此文做个总结。线程使用pthread创建(在此略过),线程会循环调用NofityDataCB函数:

static JavaVM* s_jVM = NULL;
static jobject s_jobj = NULL; //java object
static jmethodID s_jcallback = NULL; //方法id

static void NotifyDataCB(unsigned char flag, int x, int y, int w, int h, unsigned char* buff, mp_i64 timeStamp)
{
 //LOG_DBG("[NotifyDataCB()] enter.");
 JNIEnv* env;

s_jVM->AttachCurrentThread(&env, NULL); //获取当前线程的JNIEnv*

env->CallVoidMethod(s_jobj, s_jcallback, ...); //调用java层相关方法

s_jVM->DetachCurrentThread(); //释放当前线程的JNIEnv*

//LOG_DBG("[NotifyDataCB()] done.");
 return;
}

此处的重点即为:

jint AttachCurrentThread(JavaVM *vm, void **p_env, void *thr_args);

Attaches the current thread to a Java VM. Returns a JNI interface pointer in the JNIEnv argument.

其中s_jVM是由JNI_OnLoad方法保存而来的;

s_jobj在自定义的jni方法里赋值,并创建了个全局引用,使其不被虚拟机释放

s_jobj = env->NewGlobalRef(thiz); //thiz为jni方法的参数,表示java层的类对象

该s_jobj需要手动释放:

env->DeleteGlobalRef(s_jobj);

s_jcallback也是在自定义的jni方法调用以下方法保存而来:

static jmethodID GetClassMethodID(JNIEnv* env)
{
 jclass clazz = env->FindClass(classPathName); //classPathName完整的包名加类名
 if (clazz == NULL)
 {
  LOG_ERR("[GetClassMethod()]Failed to find jclass");
  return NULL; 
 }


 jmethodID jcallback = env->GetMethodID(clazz, "OnNativeDataCB", "(BIIII[IJ)V"); //获取java层方法id
 if (jcallback == NULL)
 {
  LOG_ERR("[GetClassMethod()]Failed to find method OnNativeDataCB");
  return NULL; 
 }
 
 return jcallback; //返回保存为s_jcallback
}

综上所述,关键步骤为:
1. 在onload的时候保存JavaVM指针。
2. 在自定义jni方法里(该方法须在callback方法使用前调用,例如初始化方法)保存callback方法所在对象,且该对象需要创建一个全局引用以便在线程方法里使用,默认是local ref,函数执行完会被虚拟机释放;另外自定义jni方法和callback方法在同一个类里,所以在调用自定义方法时能保存一致的jobject  。
3. 也是在自定义jni方法中,通过class获得该callback的method ID。

内容版权声明:除非注明,否则皆为本站原创文章。

转载注明出处:https://www.heiqu.com/a19b022c9d1d58e1a1703a495ad17890.html