Android JNI实现线程内部调用JAVA方法

  1. JNILib.java
class JNILib() {
    external fun start(callBack: DataCallBack):Int
    interface DataCallBack{
        fun onData(type:Int,data:ByteArray)
    }
}
  1. jni.cpp
static MessageUtil messageUtil;
// 全局变量,表示 Java 虚拟机
JavaVM* g_vm = nullptr;
// 用于保存回调对象的全局引用
jobject callBack = nullptr;
// 在 JNI 库加载时保存 Java 虚拟机的引用
extern "C" JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM* vm, void* reserved) {
    g_vm = vm;
    return JNI_VERSION_1_6;
}
extern "C"
JNIEXPORT jint JNICALL
Java_com_test_jni_JNILib_start(JNIEnv *env, jobject thiz, jobject call_back) {
    // 创建回调对象的全局引用
    callBack = env->NewGlobalRef(call_back);
    // 获取回调对象的类
    jclass callbackClass = env->GetObjectClass(callBack);
     // 启动线程并传递 Java 虚拟机、JNIEnv、回调对象及其类
    messageUtil.startThread(g_vm,env,callBack,callbackClass);
    return 0;
}
  1. message_.h
#include <jni.h>
#include <queue>
#include <pthread.h>
#include "logcat.h"
#include <string>

struct Message {
    int type;
    const char* data;
    size_t length;
};

class MessageUtil{
public:
    std::queue<Message> dataQueue;
    pthread_mutex_t mutex_;
    pthread_cond_t condition_;
    bool isThreadRunning = false;
    pthread_t msgThread;
    // 线程函数
    static void* threadFunction(void* arg);

    // 开始线程
    void startThread(JavaVM* gVm,JNIEnv* env,jobject callback,jclass callbackClass);
    // 停止线程
    void stopThread();
    // 添加数据
    void addMessage(Message msg);

    jobject callback;
    JavaVM* g_vm;
    jmethodID dataMethod;
};
  1. message.cpp
JNIEnv* AttachCurrentThreadIfNeeded(JavaVM* g_vm) {
    LogUtil::info(Message_TAG, "AttachCurrentThreadIfNeeded");
    // 声明 JNIEnv 指针
    JNIEnv* env;
    // 获取当前线程的 JNIEnv
    jint result = g_vm->GetEnv(reinterpret_cast<void**>(&env), JNI_VERSION_1_6);
    if (result == JNI_EDETACHED) {
        g_vm->AttachCurrentThread(&env, nullptr);
    }
    // 返回 JNIEnv 指针
    return env;
}
// 线程函数
void* MessageUtil::threadFunction(void* arg) {
	// 将参数转换为 MessageUtil 对象指针
    MessageUtil* messageUtil = static_cast<MessageUtil*>(arg);
    // 在当前线程附加到 Java 虚拟机
    JNIEnv* env = AttachCurrentThreadIfNeeded(messageUtil ->g_vm);
    // 线程主循环
    while (messageUtil ->isThreadRunning) {
    	// 加锁互斥量
        pthread_mutex_lock(&messageUtil ->mutex_);
		// 当数据队列为空时,线程等待条件变量
        while (messageUtil ->dataQueue.empty()) {
            pthread_cond_wait(&messageUtil ->condition_, &messageUtil ->mutex_);
        }
        // 检查线程是否需要退出
        if (!messageUtil ->isThreadRunning) {
            pthread_mutex_unlock(&messageUtil ->mutex_);
            break;
        }
        // 从队列中获取数据
        if (!messageUtil ->dataQueue.empty()){
//            LogUtil::info(Message_TAG,"成功发送数据");
            Message message = messageUtil ->dataQueue.front();
            messageUtil ->dataQueue.pop();
            // 创建字节数组并复制数据
            jbyteArray retData = env->NewByteArray(message.length);
            env->SetByteArrayRegion(retData, 0, message.length, reinterpret_cast<const jbyte*>(message.data));
            // 调用 Java 中的回调方法
            env->CallVoidMethod( messageUtil ->callback, messageUtil ->dataMethod,message.type, retData);
        }
        // 解锁互斥量
        pthread_mutex_unlock(&messageUtil ->mutex_);
    }
    // 从 Java 虚拟机中分离当前线程
    messageUtil ->g_vm->DetachCurrentThread();
    // 删除全局引用的回调对象
    env->DeleteGlobalRef(messageUtil ->callback);
    // 退出线程
    pthread_exit(NULL);
}


// 开始线程
void MessageUtil::startThread(JavaVM* gVm,JNIEnv* env,jobject callback,jclass callbackClass) {
	// 保存 Java 虚拟机、回调对象及其类
    MessageUtil::callback = callback;
    MessageUtil::g_vm = gVm;
    // 获取回调方法的 ID
    dataMethod = env->GetMethodID(callbackClass, "onData", "(I[B)V");
    isThreadRunning = true;
    // 创建线程
    pthread_create(&msgThread, NULL, &threadFunction, this);
}

// 停止线程
void MessageUtil::stopThread() {
    isThreadRunning = false;
    // 等待线程退出
    pthread_join(msgThread, NULL);
}

void MessageUtil::addMessage(Message message) {
	// 加锁互斥量
    pthread_mutex_lock(&mutex_);
    // 将消息添加到队列中
    dataQueue.push(message);
    if (dataQueue.size() > 20){
        LogUtil::info(Message_TAG,("dataQueue size = "+std::to_string(dataQueue.size())).c_str());
        dataQueue.pop();
    }
    // 发送条件信号
    pthread_cond_signal(&condition_);
    // 解锁互斥量
    pthread_mutex_unlock(&mutex_);
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/607097.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

利用大语言模型(KIMI)生成OPC UA 信息模型

在大语言模型没有出现之前&#xff0c;人们更倾向使用图形化工具或者基于窗口的软件来构建信息模型&#xff0c;图形化工具能够直观地表达信息模型中各元素之间的相互关系。但是图形化工具也有缺点&#xff0c;当描述一个复杂的信息模型时&#xff0c;图形会变得非常复杂和庞大…

如何通过OMS加快大表迁移至OceanBase

OMS&#xff0c;是OceanBase官方推出的数据迁移工具&#xff0c;能够满足众多数据迁移场景的需求&#xff0c;现已成为众多用户进行数据迁移同步的重要工具。OMS不仅支持多种数据源&#xff0c;还具备全量迁移、增量同步、数据校验等功能&#xff0c;并能够对分表进行聚合操作&…

豪投巨资,澳大利亚在追逐海市蜃楼吗?

澳大利亚政府正在积极投资于量子计算领域。继2021年向量子技术投资逾1亿澳元后&#xff0c;2023年5月&#xff0c;该国发布了首个国家量子战略&#xff0c;详细阐述了如何把握量子技术的未来及保持全球领先地位。 澳大利亚的国家量子战略概述 原文链接&#xff1a; https://ww…

jQuery-1.语法、选择器、节点操作

jQuery jQueryJavaScriptQuery&#xff0c;是一个JavaScript函数库&#xff0c;为编写JavaScript提供了更高效便捷的接口。 jQuery安装 去官网下载jQuery&#xff0c;1.x版本练习就够用 jQuery引用 <script src"lib/jquery-1.11.2.min.js"></script>…

力扣HOT100 - 4. 寻找两个正序数组的中位数

解题思路&#xff1a; 两个数组合并&#xff0c;然后根据奇偶返回中位数。 class Solution {public double findMedianSortedArrays(int[] nums1, int[] nums2) {int m nums1.length;int n nums2.length;int[] nums new int[m n];if (m 0) {if (n % 2 0) return (nums2…

游戏专用设备指纹方案解析

如同人类拥有独一无二的指纹&#xff0c;设备也有设备的指纹&#xff0c;我们可以把设备指纹理解为设备的唯一识别码。 构建设备指纹需要采集设备硬件信息、软件信息、环境信息、网络信息等维度信息&#xff0c;进行加密/压缩&#xff0c;再通过算法处理&#xff0c;赋予设备唯…

手机视频提取gif怎么操作?分享这个方法不能错过!

随着网络的发展动态gif表情包已经是人们交流的重要部分了。想要通过手机来实现视频转换gif的操作&#xff0c;还不想下载软件的情况下。可以通过使用手机端的视频转gif工具-GIF中文网&#xff0c;无需下载软件。手机端轻松一键就能在线实现视频提取gif的操作。一起来看看具体的…

【ETAS CP AUTOSAR工具链】RTA-OS基本概念与开发流程

RTA-OS基于早期ETAS操作系统的成熟技术&#xff0c;迄今为止&#xff0c;已在全球超过3.5亿个ECU中使用。RTA-OS是一个可静态配置的抢占式实时操作系统(RTOS)&#xff0c;它常被用于资源受限但有着高性能要求的方案中。内核的实现不仅遵循了AUTOSAR R3.x、R4.0、R4.1、R4.2、R4…

【stomp 实战】spring websocket 接收消息源码分析

后台消息的发送过程&#xff0c;我们通过spring websocket用户消息发送源码分析已经了解了。我们再来分析一下后端接收消息的过程。这个过程和后端发送消息过程有点类似。 前端发送消息 前端发送消息给服务端的示例如下&#xff1a; 发送给目的/app/echo一个消息。 //主动发…

英码科技推出昇腾系列AI加速卡:专为视频解析与模型推理场景打造,更具成本竞争力!

当前&#xff0c;人工智能的发展已进入加速渗透千行百业的阶段&#xff0c;算力已然成为数字化转型关键的新质生产力。随着国际挑战的加剧&#xff0c;国产算力的发展趋势愈发明显&#xff0c;市场需求也呈现出激增的态势。在这一大背景下&#xff0c;华为昇腾以其强大的技术实…

GaussianBody:基于3D高斯散射的服装人体重建

GaussianBody: Clothed Human Reconstruction via 3d Gaussian Splatting GaussianBody&#xff1a;基于3D高斯散射的服装人体重建 Mengtian Li1,2,3, Shengxiang Yao1, Zhifeng Xie1,3,2, Keyu Chen4,2, Yu-Gang Jiang2 李梦田 1,2,3 、姚胜祥 1 、谢志峰 1,3, 2 、陈科宇 4, …

谷歌开源!用 js 编写 Shell 脚本! | 开源日报 No.247

google/zx Stars: 41.4k License: Apache-2.0 zx 是一个用于编写更好脚本的工具。 提供有用的包装器&#xff0c;简化了对 child_process 的操作转义参数并提供合理的默认值使用 JavaScript 编写复杂脚本时比 Bash 更方便可以直接使用 npm 安装 dani-garcia/vaultwarden St…

长难句打卡5.9

For example, the Long Now Foundation has as its flagship project a mechanical clock that is designed to still be marking time thousands of years hence. 例如,今日永存资金会将机械钟表视为旗舰项目,因此该钟表旨在为未来几千年保持计时。 Foundation n.基金会flag…

数据库(MySQL)—— 索引

数据库&#xff08;MySQL&#xff09;—— 索引 什么是索引创建索引使用 CREATE INDEX 语句使用 ALTER TABLE 语句在创建表时定义索引特殊类型索引注意事项 举个例子无索引的情况有索引的情况为什么索引快索引的结构 今天我们来看看MySQL中的索引&#xff1a; 什么是索引 MyS…

unity基础(一)

内容概要&#xff1a; 生命周期函数vector3 位置 方向 缩放旋转等信息Vector3欧拉角和Quaternion四元素unity脚本执行顺序设置 一 生命周期函数 方法说明Awake最早调用,所以一般可以再此实现单例模式OnEnable组件激活后调用,在Awake后会调用一次Start在Update之前调用一次&a…

硬件知识积累 音频插座的了解,看音频插座的原理图来了解音频插座的引脚。

1. 音频接口 音频插座是一种用于连接音频信号线路的电子元件&#xff0c;常见于音频设备&#xff08;如音响、耳机、话筒等&#xff09;中。它的主要作用是将电子信号转化为声音信号&#xff0c;以满足人们对于音乐、电影、游戏等方面的需求。 根据插头形状的不同&#xff0c;音…

和comate一起,用JavaScript实现一个简易版五子棋小游戏

前言 五子棋起源于中国&#xff0c;是全国智力运动会竞技项目之一&#xff0c;是一种两人对弈的纯策略型棋类游戏。双方分别使用黑白两色的棋子&#xff0c;下在棋盘直线与横线的交叉点上&#xff0c;先形成五子连珠者获胜。 这次和Baidu Comate智能代码助手共同完成这个小游戏…

[华为OD] C卷 田忌赛马 DFS 200

题目&#xff1a; 给定两个只包含数字的数组a, b,调整数组a里面数字的顺序&#xff0c;使得尽可能多的a[i] >b[i]。 数组a和b中的数字各不相同。 输出所有可以达到最优结果的a数组的数量 输入描述 输入的第一行是数组a中的数字&#xff0c;其中只包含数字&#xff0c;每…

LVS DR模式部署

一、LVS 简介 LVS的三种工作模式 NAT 地址转换 调度器会作为所有节点服务器的默认网关&#xff0c;也是客户端的访问入口和节点服务器返回响应消息的出口&#xff0c;所以调度器会承载双向流量的负载压力&#xff0c;可能会为整个群集的性能瓶颈。由于节点服务器都会处于内网…

AcWing 4993 FEB

4993. FEB - AcWing题库 大佬亲笔 将原串分成三段&#xff1a; FFF|E.....B|FFF 先合并中间段&#xff0c;再合并两边的段 #include <iostream> #include <cstring> #include <algorithm> #include <string> #include <queue&g…