Android JNI 在C++多线程调用Java的回调函数

java 接口定义:

java 复制代码
public interface Callback {
    void onResult(int status, String message);
}

java 设置回调:

java 复制代码
public class Sample {
    static {
        System.loadLibrary("sample_core_jni");
    }

    public static native int process(long handle, String message);


    public static void onCallback(int status, String message) {
        if (mCallback != null) {
            mCallback.onResult(status, message);
        }
    }

    private static Callback mCallback;

    public static void setCallback(Callback callback) {
        if (callback == null) {
            throw new NullPointerException("callback can not be null");
        }
        mCallback = callback;
    }

// 添加其他native函数
....
}

JNI 注册回调函数:

cpp 复制代码
#include <jni.h>
#include <iostream>
 
// 全局引用,确保JNI环境可以访问
JavaVM *g_VM;
jclass gCallbackClassSample = nullptr;
jmethodID gOnCallbackMethodID = nullptr;


JNIEXPORT jint JNI_OnLoad(JavaVM *vm, void *reserved) {
    JNIEnv *env = nullptr;
    jint result = -1;
    if (vm->GetEnv((void **) &env, JNI_VERSION_1_6) != JNI_OK) {
        return result;
    }

    //JavaVM是虚拟机在JNI中的表示,等下再其他线程回调java层需要用到
    env->GetJavaVM(&g_VM);

    register_fields(env);

    LOGI("JNI OnLoad result : %d", result);
    return JNI_VERSION_1_6;
}

register_fields(JNIEnv *env) {

//注册Java回调函数所在的类及回调函数
}



int ChatCallback(int status, const char *output) {
    JNIEnv *env;
    bool mNeedDetach = false;
    //获取当前native线程是否有没有被附加到jvm环境中
    int getEnvStat = (*g_VM).GetEnv( (void **)&env, JNI_VERSION_1_6);
    if (getEnvStat == JNI_EDETACHED) {
        //如果没有, 主动附加到jvm环境中,获取到env
        if (g_VM->AttachCurrentThread(&env, NULL) != JNI_OK) {
            LOGE("ChatCallback failed to attach current thread");
        }
        mNeedDetach = JNI_TRUE;
    }
    jString j_out = native_message_to_java_message(env, output);

    if (fields.chat_result_callback != nullptr) {
        env->CallStaticVoidMethod(gCallbackClassSample, gOnCallbackMethodID, (int)status, j_out);
    }
    //释放当前线程
    if(mNeedDetach) {
        g_VM->DetachCurrentThread();
    }
    env = NULL;

    return 0;
}

extern "C"
JNIEXPORT void JNICALL
Java_com_example_yourapp_process(JNIEnv *env, jlong handle, jString message) {

     //
     const char* c_message = ...
     //该函数为异步函数
     int err = Chat(handle, c_message, ChatCallback);
}
相关推荐
考虑考虑6 小时前
Mybatis实现批量插入
java·后端·mybatis
咖啡八杯7 小时前
GoF设计模式——中介者模式
java·后端·spring·设计模式
青石路11 小时前
记一次多JDK版本问题的排查,一坑套一坑,差点没爬上来
java
见过夏天12 小时前
C++ 基础入门完全指南
c++
杉氧13 小时前
深入理解 Compose 重组机制:快照系统如何驱动 UI 精准刷新?
android·架构·android jetpack
召钱熏13 小时前
状态枚举正确≠渲染正确:一个语音按钮的状态机边界修复实录
android·前端
像我这样帅的人丶你还14 小时前
Java 后端详解(五):Redis 缓存
java·后端·全栈
杉氧14 小时前
深度解析:Jetpack Compose 核心架构与底层原理 —— 十年安卓老兵的“破茧重生”
android·架构·android jetpack
通玄14 小时前
Jetpack Compose 入门系列(七):ViewModel 与界面状态管理
android
落魄Android在线炒饭14 小时前
Android Framework 开发技巧:android.jar 生成与系统快速编译验证
android