Android - NDK :JNI实现异步回调

在android代码中,通过JNI调用c层子线程执行耗时任务,在c层子线程中把结果回调到android层,
C语言小白,请批评指正!
使用场景

android层代码:


import androidx.appcompat.app.AppCompatActivity;

import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Toast;

import com.alibaba.fastjson.JSON;
import com.hisign.callback.MyCallback;
import com.hisign.callback.MyCallbackManager;
import com.hisign.device.databinding.ActivityMainDeviceBinding;

public class MainActivity extends AppCompatActivity {
    static {
        System.loadLibrary("broadcast");
    }

    private ActivityMainDeviceBinding binding;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        binding = ActivityMainDeviceBinding.inflate(getLayoutInflater());
        setContentView(binding.getRoot());

        binding.btnAsync.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                MyCallbackManager.getInstance().doAsyncWork(new MyCallback() {
                    @Override
                    public void onStart() {
                        runOnUiThread(() -> Toast.makeText(MainActivity.this, "onStart...", Toast.LENGTH_SHORT).show());
                    }

                    @Override
                    public void onResult(String result) {
                        runOnUiThread(() -> binding.tvDev.setText(result));
                    }

                    @Override
                    public void onEnd() {
                        runOnUiThread(() -> {
                            Toast.makeText(MainActivity.this, "onEnd !!!", Toast.LENGTH_SHORT).show();
                            Log.d("lixm", "run: onEnd...");
                        });
                    }
                });
            }
        });
    }
}

activity_main_device.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity"
    android:orientation="vertical">

    <TextView
        android:id="@+id/sample_text"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Hello World!"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

    <Button
        android:id="@+id/btnAsync"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="20dp"
        android:text="异步回调"
        tools:ignore="MissingConstraints" />

    <TextView
        android:id="@+id/tvDev"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="20dp"
        android:gravity="center_horizontal"
        android:textSize="40sp"
        android:textStyle="bold"/>
</LinearLayout>

MyCallback.java

public interface MyCallback {
    public void onStart();
    public void onResult(String result);
    public void onEnd();
}

MyCallbackManager.java


public class MyCallbackManager {
    private static MyCallbackManager instance;
    private MyCallbackManager(){
    }

    public static MyCallbackManager getInstance(){
        if(instance == null){
            synchronized (MyCallbackManager.class){
                if(instance == null){
                    instance = new MyCallbackManager();
                }
            }
        }
        return instance;
    }

    public native void doAsyncWork(MyCallback callback);
}


jni代码

thread_callback.cpp

#include "callback/ThreadCallback.h"
#include "iostream"
#include <jni.h>
#include "jniLog.h"
using namespace std;

// 在线程里面调用的,都需要声明为全局变量
JavaVM *jvm;
jobject mCallBackObj;

jmethodID resultMid;
jmethodID endMid;
JNIEnv *mEnv;
int mIsDetach = JNI_FALSE;

extern "C"
JNIEXPORT void JNICALL
Java_com_hisign_callback_MyCallbackManager_doAsyncWork(JNIEnv *env, jobject thiz,
        jobject callback) {
    LOGI("native _doAsyncWork...");

    //JavaVM是虚拟机在JNI中的表示,等下再其他线程回调java层需要用到
    env->GetJavaVM(&jvm);
    // 三步曲
    // 1 获取java层的类对象
    mCallBackObj = env->NewGlobalRef(callback);  // 创建一个jni全局引用
    jclass clz = env->GetObjectClass(callback);
    // 2 获取java层的函数
    jmethodID startMid = env->GetMethodID(clz,"onStart", "()V");
    // 这两个方法因为要在子线程中回调,所以声明为全局的
    resultMid = env->GetMethodID(clz,"onResult", "(Ljava/lang/String;)V");
    endMid = env->GetMethodID(clz,"onEnd", "()V");
    // 3 调用函数
    env->CallVoidMethod(mCallBackObj,startMid);

    MyCallback myCallback = [](std::string ret){  // 子线程中计算结果的回调
        //获取当前native线程是否有没有被附加到jvm环境中
        int envStatus = jvm->GetEnv((void **) &mEnv, JNI_VERSION_1_6);
        if(envStatus == JNI_EDETACHED){
            //如果没有, 主动附加到jvm环境中,获取到mEnv
            if (jvm->AttachCurrentThread(&mEnv, NULL) != 0) {
                return;
            }
            mIsDetach = JNI_TRUE;
        }
        // 回调数据到java层
        jstring jstr = mEnv->NewStringUTF(ret.c_str());         // 生成java层的String(jstring)
        mEnv->CallVoidMethod(mCallBackObj, resultMid, jstr);    // 回调到java层
    };
    CallbackEnd cbend = [](){   // 任务完成的回调
        if(mIsDetach){
            mEnv->CallVoidMethod(mCallBackObj,endMid);
            //释放当前线程
            jvm->DetachCurrentThread();
            mEnv = NULL;
        }
    };
    doAsyncWork(myCallback,cbend);
}


ThreadCallback.h

//
// Created by lixm on 2025/1/6.
//
#ifndef BROADCASTNATIVE_THREADCALLBACK_H
#define BROADCASTNATIVE_THREADCALLBACK_H

#include <functional>
typedef std::function<void(std::string)> MyCallback;
typedef std::function<void()> CallbackEnd;
void doAsyncWork(MyCallback MyCallback,CallbackEnd callbackEnd);
#endif //BROADCASTNATIVE_THREADCALLBACK_H

ThreadCallback.cpp

#include <thread>
#include "ThreadCallback.h"
#include "iostream"
#include "../jniLog.h"
#include "unistd.h"
using namespace std;


/**
 * 执行耗时操作
 * @param callback  异步回调
 */
void AsyncWork(MyCallback mycb,CallbackEnd callbackEnd){
    int i = 10;
    while(i -- > 0){
        sleep(1);  // 休眠1秒,模拟耗时操作(unistd.h)
        mycb(to_string(i));
    }
    callbackEnd();
}

void doAsyncWork(MyCallback myCallback,CallbackEnd callbackEnd){
    /**
     * 这里有三个参数
     * 第一个参数是子线程执行的函数
     * 第二个参数是函数的参数(回调计算结果)
     * 第三个参数是函数的参数(回调计算完成)
     */
    thread t1(AsyncWork,myCallback,callbackEnd);
    //t1.join();  // 主线程阻塞,等待线程执行完成
    t1.detach();
}

CmakeLists.txt

cmake_minimum_required(VERSION 3.18.1)

project("broadcast")
add_library( # Sets the name of the library.
        broadcast
        SHARED
        thread_callback.cpp
        callback/ThreadCallback.cpp)

find_library(log-lib log)
target_link_libraries(broadcast ${log-lib})

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

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

相关文章

Java Web开发进阶——RESTful API设计与开发

随着分布式系统和微服务架构的流行&#xff0c;RESTful API已成为现代Web应用中后端与前端、第三方系统交互的重要方式。本节将深入探讨RESTful API的设计原则、实现方式以及如何使用Spring Boot开发高效、可靠的RESTful服务。 1. 理解RESTful API的设计原则 1.1 什么是RESTfu…

PWR-STM32电源控制

一、原理 睡眠模式不响应其他操作&#xff0c;比如烧写程序&#xff0c;烧写时按住复位键松手即可下载&#xff0c;在禁用JTAG也可如此烧写程序。 对于低功耗模式可以通过RTC唤醒、外部中断唤醒、中断唤醒。 1、电源框图&#xff1a; VDDA主要负责模拟部分的供电、Vref和Vref-…

深兰科技董事长陈海波应邀为华东师大心理学专业师生做AI专题讲座

12月28日&#xff0c;应上海华东师范大学的邀请&#xff0c;上海市科协常委、上海交通大学博士生导师、深兰科技创始人兼董事长陈海波专程到校&#xff0c;为该校心理学专业的全体师生做了一场关于人工智能推动个人数字化未来的专题讲座。 他在演讲中&#xff0c;首先详细讲述了…

ssh2-sftp-client uploadDir Upload error: getLocalStatus: Bad path: ./public

报错解释 这个错误表明在使用 ssh2-sftp-client 这个Node.js库进行目录上传时遇到了问题。具体来说&#xff0c;是指定的本地路径&#xff08;./public&#xff09;不正确或者不存在。 解决方法&#xff1a; 确认当前工作目录&#xff1a;确保你在执行上传操作时的当前工作目…

Vue指令的综合案例

Vue指令的综合案例 参考文献&#xff1a; Vue的快速上手 Vue指令上 Vue指令下 文章目录 Vue指令的综合案例记事本 列表渲染删除功能添加功能底部统计和清空总代码 结语 博客主页: He guolin-CSDN博客 关注我一起学习&#xff0c;一起进步&#xff0c;一起探索编程的无限可能…

2025新春烟花代码(二)HTML5实现孔明灯和烟花效果

效果展示 源代码 <!DOCTYPE html> <html lang"en"> <script>var _hmt _hmt || [];(function () {var hm document.createElement("script");hm.src "https://hm.baidu.com/hm.js?45f95f1bfde85c7777c3d1157e8c2d34";var …

# 网络编程 - 轻松入门不含糊

网络编程 - 轻松入门 介绍 网络编程指的是&#xff0c;在网络通信协议下。实现 不同计算机之间 的数据传输&#xff0c;例如 通信、聊天、视频通话 等。 1. 网络编程概述 学习网络编程过程 中我们可以将网络编程理解为 计算机之间的数据交互 但 前提是通…

SAP BC 同服务器不同client之间的传输SCC1

源配置client不需要释放 登录目标client SCC1

【大数据基础】大数据概述

【作者主页】Francek Chen 【专栏介绍】 ⌈ ⌈ ⌈大数据技术原理与应用 ⌋ ⌋ ⌋专栏系统介绍大数据的相关知识&#xff0c;分为大数据基础篇、大数据存储与管理篇、大数据处理与分析篇、大数据应用篇。内容包含大数据概述、大数据处理架构Hadoop、分布式文件系统HDFS、分布式数…

【ROS2】☆ launch之Python

☆重点 ROS1和ROS2其中一个很大区别之一就是launch的编写方式。在ROS1中采用xml格式编写launch&#xff0c;而ROS2保留了XML 格式launch&#xff0c;还另外引入了Python和YAML 编写方式。选择哪种编写取决于每位开发人员的爱好&#xff0c;但是ROS2官方推荐使用Python方式编写…

Shell编程详解

文章目录 一、Linux系统结构二、Shell介绍1、Shell简介2、Shell种类3、Shell查询和切换 三、Shell基础语法1、注释2、本地变量3、环境变量3.1、查看环境变量3.2、临时设置环境变量3.3、永久设置环境变量 4、特殊变量5、控制语句5.1、shell中的中括号5.2、if语句5.3、for循环5.4…

Zemax 序列模式下的扩束器

扩束器结构原理 扩束器用于增加准直光束&#xff08;例如激光束&#xff09;的直径&#xff0c;同时保持其准直。它通常用于激光光学和其他需要修改光束大小或发散度的应用。 在典型的扩束器中&#xff0c;输入光束是准直激光器&#xff0c;或光束进入第一个光学元件。当光束开…

react-quill 富文本组件编写和应用

index.tsx文件 import React, { useRef, useState } from react; import { Modal, Button } from antd; import RichEditor from ./RichEditor;const AnchorTouchHistory: React.FC () > {const editorRef useRef<any>(null);const [isModalVisible, setIsModalVis…

【深度学习】多目标融合算法(二):底部共享多任务模型(Shared-Bottom Multi-task Model)

目录 一、引言 1.1 往期回顾 1.2 本期概要 二、Shared-Bottom Multi-task Model&#xff08;SBMM&#xff09; 2.1 技术原理 2.2 技术优缺点 2.3 业务代码实践 三、总结 一、引言 在朴素的深度学习ctr预估模型中&#xff08;如DNN&#xff09;&#xff0c;通常以一个行…

探秘MetaGPT:革新软件开发的多智能体框架(22/30)

一、MetaGPT 引发的 AI 变革浪潮 近年来&#xff0c;人工智能大模型领域取得了令人瞩目的进展&#xff0c;GPT-3、GPT-4、PaLM 等模型展现出了惊人的自然语言处理能力&#xff0c;仿佛为 AI 世界打开了一扇通往无限可能的大门。它们能够生成流畅的文本、回答复杂的问题、进行创…

LabVIEW软件Bug的定义与修改

在LabVIEW软件开发过程中&#xff0c;bug&#xff08;程序错误或缺陷&#xff09;指的是程序中导致不符合预期行为的任何问题。Bug可能是由于编码错误、逻辑漏洞、硬件兼容性问题、系统资源限制等因素引起的。它可能会导致程序崩溃、功能无法正常执行或输出结果不符合预期。理解…

高性能网络模式:Reactor 和 Proactor

Reactor Reactor 采用I/O多路复用监听事件&#xff0c;收到事件后&#xff0c;根据事件类型分配给某个进程/线程。 实际应用中用到的模型&#xff1a; 单 Reactor 单进程 单 Reactor 多线程 优点&#xff1a;能充分利用多核CPU性能。 缺点&#xff1a;存在多线程竞争共享资源…

扩散模型论文概述(三):Stability AI系列工作【学习笔记】

视频链接&#xff1a;扩散模型论文概述&#xff08;三&#xff09;&#xff1a;Stability AI系列工作_哔哩哔哩_bilibili 本期视频讲的是Stability AI在图像生成的工作。 同样&#xff0c;第一张图片是神作&#xff0c;总结的太好了&#xff01; 介绍Stable Diffusion之前&…

大数据技术-Hadoop(四)Yarn的介绍与使用

目录 一、Yarn 基本结构 1、Yarn基本结构 2、Yarn的工作机制 二、Yarn常用的命令 三、调度器 1、Capacity Scheduler&#xff08;容量调度器&#xff09; 1.1、特点 1.2、配置 1.2.1、yarn-site.xml 1.2.2、capacity-scheduler.xml 1.3、重启yarn、刷新队列 测试 向hi…

玩转大语言模型——ollama导入huggingface下载的模型

ollama导入huggingface模型 前言gguf模型查找相关模型下载模型 导入Ollama配置参数文件导入模型查看导入情况 safetensfors模型下载模型下载llama.cpp配置环境并转换 前言 ollama在大语言模型的应用中十分的方便&#xff0c;但是也存在一定的问题&#xff0c;比如不能使用自己…