JNI回调用中不同线程的env无法找到正确的kotlin的class

不同线程都需要通过 JavaVM 获取到的 JNIEnv 指针, 如果有两个线程有两个 env。 其中一个是jni接口自己传过来的,可以正常使用,正常获取kotlin中的class。但是通过 JavaVM 新获取的env 无法找到kotlin的class

1. 确保线程已附加到 JVM

确保每个线程在使用 JNIEnv 之前已附加到 JVM。可以在每个线程中调用 AttachCurrentThread

cpp 复制代码
JavaVM* g_jvm = nullptr;

JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM* vm, void* reserved) {
    g_jvm = vm;
    return JNI_VERSION_1_6;
}

void someJNIFunction() {
    JNIEnv* env;
    if (g_jvm->AttachCurrentThread(&env, nullptr) != JNI_OK) {
        // 处理错误
        return;
    }

    // 使用 env 进行 JNI 调用
    jclass cls = env->FindClass("com/example/YourKotlinClass");
    if (cls == nullptr) {
        // 处理类未找到的错误
    }

    // 线程使用完成后,分离线程
    g_jvm->DetachCurrentThread();
}

2. 确保正确缓存类引用(关键:类要转化成全局引用即可给不同env使用)

确保类引用被正确缓存,并且在不同的线程中可用。可以在 JNI_OnLoad 中缓存类引用:

cpp 复制代码
jclass g_yourKotlinClass; //这里是关键

JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM* vm, void* reserved) {
    g_jvm = vm;
    JNIEnv* env;
    if (vm->GetEnv(reinterpret_cast<void**>(&env), JNI_VERSION_1_6) != JNI_OK) {
        return JNI_ERR;
    }

		// 这部分代码也可以写在 someJNIFunction 内。 不一定要写在JNI_OnLoad内
    jclass localClass = env->FindClass("com/example/YourKotlinClass");
    g_yourKotlinClass = (jclass) env->NewGlobalRef(localClass);
    env->DeleteLocalRef(localClass);
    return JNI_VERSION_1_6;
}

3. 例子:完整示例代码

cpp 复制代码
#include <jni.h>
#include <android/log.h>

#define LOG_TAG "JNI_OnLoad"
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__)

JavaVM* g_jvm = nullptr;
jclass g_yourKotlinClass = nullptr;

JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM* vm, void* reserved) {
    g_jvm = vm;
    JNIEnv* env;
    if (vm->GetEnv(reinterpret_cast<void**>(&env), JNI_VERSION_1_6) != JNI_OK) {
        return JNI_ERR;
    }

    jclass localClass = env->FindClass("com/example/YourKotlinClass");
    if (localClass == nullptr) {
        return JNI_ERR; // 类加载失败
    }

    g_yourKotlinClass = (jclass) env->NewGlobalRef(localClass);
    env->DeleteLocalRef(localClass);
    return JNI_VERSION_1_6;
}

void someJNIFunction() {
    JNIEnv* env;
    if (g_jvm->AttachCurrentThread(&env, nullptr) != JNI_OK) {
        // 处理错误
        return;
    }

    if (g_yourKotlinClass == nullptr) {
        LOGD("Kotlin class is null");
        return;
    }

    jmethodID methodID = env->GetMethodID(g_yourKotlinClass, "yourMethodName", "()V");
    if (methodID == nullptr) {
        LOGD("Method ID not found");
        return;
    }

    jobject obj = ... // 获取对象实例
    env->CallVoidMethod(obj, methodID);

    g_jvm->DetachCurrentThread();
}
相关推荐
yaoxin5211234 小时前
434. Java 日期时间 API - Period 基于日期的时间段
java·开发语言·python
凡人叶枫5 小时前
Effective C++ 条款30:透彻了解 inlining 的里里外外
linux·开发语言·c++·嵌入式开发·effective c++
noipp5 小时前
推荐题目:洛谷 P10907 [蓝桥杯 2024 国 B] 蚂蚁开会
c语言·c++·算法·编程·洛谷
学逆向的5 小时前
C++纯虚函数
开发语言·c++·网络安全
程序员二叉5 小时前
【JUC】ThreadLocal底层原理|内存泄漏|弱引用|跨线程传递方案
java·开发语言·面试·职场和发展·juc
程序员二叉5 小时前
【JUC】线程池全套深度详解|参数|流程|拒绝策略|调优|异常处理
java·开发语言·jvm·算法·面试·juc
凡人叶枫6 小时前
Effective C++ 条款22:将成员变量声明为 private
linux·开发语言·c++
Qt程序员6 小时前
掌握 Linux 内核调度:从原理到实现(进程篇)
java·开发语言
code bean6 小时前
【LangChain】检索器完全指南:从向量检索到生产级 RAG 架构
java·开发语言·微服务
LabVIEW开发7 小时前
LabVIEW + MATLAB 混合编程:爆炸场测试数据精准采集方案
开发语言·matlab·labview