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);
}
相关推荐
我不想当小卡拉米几秒前
C++:继承+菱形虚拟继承的一箭双雕
开发语言·jvm·c++
ChinaRainbowSea15 分钟前
1. 初始 RabbitMQ 消息队列
java·中间件·rabbitmq·java-rabbitmq
机器视觉知识推荐、就业指导18 分钟前
QML 批量创建模块 【Repeater】 组件详解
前端·c++·qml
lmryBC4924 分钟前
golang接口-interface
java·前端·golang
ゞ 正在缓冲99%…25 分钟前
leetcode75.颜色分类
java·数据结构·算法·排序
橘猫云计算机设计37 分钟前
基于springboot的考研成绩查询系统(源码+lw+部署文档+讲解),源码可白嫖!
java·spring boot·后端·python·考研·django·毕业设计
时光呢41 分钟前
JAVA常见的 JVM 参数及其典型默认值
java·开发语言·jvm
橙橙子23043 分钟前
c++柔性数组、友元、类模版
开发语言·c++·柔性数组
程序媛学姐1 小时前
SpringKafka错误处理:重试机制与死信队列
java·开发语言·spring·kafka
向阳2561 小时前
SpringBoot+vue前后端分离整合sa-token(无cookie登录态 & 详细的登录流程)
java·vue.js·spring boot·后端·sa-token·springboot·登录流程