JNI:在线程或信号处理函数中访问自定义类

时间:2023-03-09 17:55:37
JNI:在线程或信号处理函数中访问自定义类
在写一个Tomcat应用,类需要被信号处理函数回调,可是在单独的程序中测试没用问题:
void OnSingalHandler(int sig)
{
...
JNIEnv* env=NULL;
if (g_VM->AttachCurrentThread((void**)&env, NULL) != JNI_OK)
{
printf("AttachCurrentThread failed!\n");
} jclass cls = env->FindClass("nms/scada/Rtdb");
jmethodID callback = env->GetStaticMethodID(cls,"RealDataChange","(Ljava/lang/String;)V");
env->CallStaticVoidMethod(cls,callback,CStr2Jstring(env,xmlDoc.GetDoc().c_str())); g_VM->DetachCurrentThread(); ... }

    一切安好,可是移到Tomcat中时,FindClass就不工作了,无论如何返回都是NULL,看网上说是因为Tomcat的ClassLoader不一样了,种种,试了好久未果。

    后来发现即使是在Tomcat中,在JNI_OnLoad函数依然可以用FindClass调用成功,唯独在线程或者信号处理函数中有问题,或许是和AttachCurrentThread有关系吧。

    于是想用全局变量缓存JNI_OnLoad中FindClass的结果,结果失败。

后来又有人说全局变量缓存是可以用的,不过要用NewGlobalRef创建全局对象引用,查了一下NewGlobalRef的定义:

   jobject NewGlobalRef(jobject obj);

根本就没有给jclass用的啊,于是又开始乱找原因,试着用Tomcat的ClassLoader什么的原因。最终也没有成功。

又不甘心,查看jobject和jclass尼玛,原来是一回事

     typedef jobject jclass;

好了,一用,妥妥的成功了。

JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *vm, void *reserved)
{
...
jclass classClass = env->FindClass("nms/scada/Rtdb");
g_Class = (jclass)env->NewGlobalRef((jobject)classClass);
...
}

  完。