Android JNI复杂用法,回调,C++中调用Java方法

Android JNI复杂用法,回调,C++中调用Java方法

一、前言

Android JNI的 普通用法估计很多人都会,但是C++中调用Java方法很多人不熟悉,并且网上很多介绍都是片段的。

虽然C/C++调用Java不常用,但是掌握多一点还是有好处的。

Android JNI的基础知识介绍,之前已经有介绍,不熟悉的可以先看看:

Android Jni的介绍和简单Demo实现:

https://blog.csdn.net/wenzhi20102321/article/details/136291126

本文主要介绍JNI C++调用Java代码实现和相关知识,有兴趣的可以看看。

二、C++调用Java方法实现代码

1、上层代码 MainAcitvity.java

package com.demo.jnicallback;import androidx.appcompat.app.AppCompatActivity;import android.os.Bundle;
import android.util.Log;
import android.widget.TextView;public class MainActivity extends AppCompatActivity {String TAG = "MainActivity.java";static {System.loadLibrary("native-lib");}@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);Log.i(TAG, "conCreate");TextView tv = findViewById(R.id.sample_text);String jniString = stringFromJNI();Log.i(TAG, "conCreate cppCallBackMethod jniString = " + jniString);tv.setText("" + jniString);}//C++调用Java 的方法,定义成private方法,cpp也是可以调用到的,因为是通过反射过来的public void cppCallBackMethod(String name, int age) {Log.i(TAG, "cppCallBackMethod name = " + name + ",age = " + age);}//Java 调用到 cpp 的native方法public native String stringFromJNI();}

布局上未做修改,运行后的默认字符串"Hello from C++"。

Java代码这里加了一个给C++调用过来的方法,具体实现效果可以看是日志。

2、cpp代码 native-lib.cpp 代码:

#include <jni.h>
#include <string>#include <android/log.h> //添加头文件
#define LOG_TAG "native-lib.cpp" //定义TAG
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO,LOG_TAG,__VA_ARGS__)
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__)
#define LOGW(...) __android_log_print(ANDROID_LOG_WARN, LOG_TAG, __VA_ARGS__)
#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR,LOG_TAG,__VA_ARGS__)#include <iostream>
#include <chrono>
#include <thread>extern "C" JNIEXPORT jstring JNICALL
Java_com_demo_jnicallback_MainActivity_stringFromJNI(JNIEnv* env,jobject thiz /* this */) {std::string hello = "Hello from C++";LOGI("stringFromJNI hello = %s", hello.c_str());//c++调用Java方法:public void cppCallBackMethod(String name, int age)jobject m_object = env->NewGlobalRef(thiz);//创建对象的本地变量jclass mainActivityCls=env->FindClass("com/demo/jnicallback/MainActivity");//获取类对象jmethodID cppCallBackMethod = env->GetMethodID(mainActivityCls, "cppCallBackMethod", "(Ljava/lang/String;I)V");const char *message = "cppA";int age = 10;env->CallVoidMethod(m_object, cppCallBackMethod, env->NewStringUTF(message), age);while (age < 50) {//睡眠1秒std::chrono::seconds duration(1); // 休眠一秒钟std::this_thread::sleep_for(duration);age = age + 10;env->CallVoidMethod(m_object, cppCallBackMethod, env->NewStringUTF(message), age);}return env->NewStringUTF(hello.c_str());
}

上面代码可以看到获取类对象,是为了获取方法id;获取对象的本地变量是为了调用方法。

网上有些示例可能写法不一样,熟悉c++代码的应该知道,"env->“的写法和”(*env)."是一个意思。

3、效果日志:

//Java打印最开始日志
2024-03-01 16:27:50.401  I/MainActivity.java: conCreate
//cpp文件打印,开始的日志
2024-03-01 16:27:50.402  I/native-lib.cpp: stringFromJNI hello = Hello from C++
//cpp调用Java部分日志,在Java代码每隔一秒的打印
2024-03-01 16:27:50.402  I/MainActivity.java: cppCallBackMethod name = cppA,age = 10
2024-03-01 16:27:51.403  I/MainActivity.java: cppCallBackMethod name = cppA,age = 20
2024-03-01 16:27:52.403  I/MainActivity.java: cppCallBackMethod name = cppA,age = 30
2024-03-01 16:27:53.403  I/MainActivity.java: cppCallBackMethod name = cppA,age = 40
2024-03-01 16:27:54.404  I/MainActivity.java: cppCallBackMethod name = cppA,age = 50
//Java onCreate最后的日志,打印C++返回的字符串
2024-03-01 16:27:54.404  I/MainActivity.java: conCreate cppCallBackMethod jniString = Hello from C++

上面的代码就有Java --> C++和C++ --> Java的代码流程。

注意,这里的示例代码添加了睡眠代码,如果在主线程长时间执行任务是有可能导致ANR的。

4、cpp代码 native-lib.cpp 代码另一种写法

下面这种写法不用NewGlobalRef创建对象的本地变量。

中间的区别就是这里函数的调用没有使用"->“,使用的”(*env)."

extern "C" JNIEXPORT jstring JNICALL
Java_com_demo_jnicallback_MainActivity_stringFromJNI(JNIEnv* env,jobject thiz /* this */) {std::string hello = "Hello from C++";LOGI("stringFromJNI hello = %s", hello.c_str());//c++调用Java方法:public void cppCallBackMethod(String name, int age)jclass mainActivityCls=(*env).FindClass("com/demo/jnicallback/MainActivity");//获取类对象jmethodID cppCallBackMethod = (*env).GetMethodID(mainActivityCls, "cppCallBackMethod", "(Ljava/lang/String;I)V");const char *message = "cppA";int age = 10;(*env).CallVoidMethod(thiz, cppCallBackMethod, env->NewStringUTF(message), age);while (age < 50) {//睡眠1秒std::chrono::seconds duration(1); // 休眠一秒钟std::this_thread::sleep_for(duration);age = age + 10;(*env).CallVoidMethod(thiz, cppCallBackMethod, env->NewStringUTF(message), age);}return env->NewStringUTF(hello.c_str());
}

上面的代码运行也是一样的效果。

"->“和”(*env)."有啥区别?因为不是很熟悉,还还说不清。

上面不同写法调用方法的参数是有区别的,其实就是函数api的参数要求不同,具体可以看到jni.h的源码。

三、其他

1、C++到Java 相关api函数介绍

上面示例中使用用到的api:

(1)jobject m_object = env->NewGlobalRef(thiz);//创建对象的本地变量
(2)jclass mainActivityCls=env->FindClass("com/demo/jnicallback/MainActivity");//获取类对象(3)jmethodID cppCallBackMethod = env->GetMethodID(mainActivityCls, "cppCallBackMethod", "(Ljava/lang/String;I)V");(4)env->CallVoidMethod(m_object, cppCallBackMethod, env->NewStringUTF(message), age);

上面(1)和(2)是没什么研究价值的,NewGlobalRef和FindClass都是固定的写法。

(3)和(4)的不用方法的调用区别就比较大了,使用不同的api函数还可以修改Java的变量属性。

静态方法和动态方法调用的api函数也不一样,有返回值的方法和没有返回值的方法调用的api函数也是不一样的。

并且Java方法或者变量即使是private修饰的也不影响cpp调用过去,因为反射是不受修饰符影响的。

第3步里面的签名字符串“(Ljava/lang/String;I)V”,表示的是Java的方法和返回值的签名,唯一性;

这里面的签名字符串都是根据Java方法和方法的参数进行变化的。

下面对3、4步的代码相关知识做展开介绍。

2、调用获取不同方法和变量的api

方法、变量修饰类型表格
函数描述描述
GetFieldID得到一个实例的域的ID
GetStaticFieldID得到一个静态的域的ID
GetMethodID得到一个实例的方法的ID
GetStaticMethodID得到一个静态方法的ID

上面Jni.cpp调用Java代码已经用到部分api方法,并且从字面含义也是比较容易里面这个表格的api的具体作用。

这个表格的用于就是为了获取到方法的修饰类型,比如方法,静态方法,变量,静态变量。

毕竟不同的修饰类型,在编译过程是有差异的。所以要区分。

3、Java签名类型字符串 常用的数据类型及对应字符:

上面示例中的"(Ljava/lang/String;I)V");字符串都是根据Java的方法通过下面这个表格转换来的。

Java 类型Jni中表示的符号备注
booleanZ不是类型首字母大写
byteB
charC
shortS
intI
longL
floatF
doubleD
voidV
objects对象Lfully-qualified-class-name;L全类名;记得最后是有分号的
Arrays数组[array-type [数组类型
methods方法(argument-types)return-type(参数类型)返回类型

这个表格是有有啥用?就更多人懵逼了。

其实这些类型符号表示的是Java方法或者属性的一个签名,唯一性,目前就是为了让Jni.cpp调用到Java代码。

举个例子就很容易清楚了:

//XXX.Javaint age;String name;public  int add(int number1,int number2){System.out.println("c/C++居然调用了我");return number1+number2;}//jni.cpp 修改Java属性值和调用Java方法示例//获取类对象
jclass mainActivityCls=env->FindClass("com/zmw/jnitest/MainActivity");//获取属性的fieldId,--》这里就用到了签名类型
jfieldID ageFid = env->GetFieldID(mainActivityCls,"age","I");
jfieldID nameFid=env->GetFieldID(mainActivityCls, "name", "Ljava/lang/String;");
//获取属性值
jint  age = env->GetIntField(mainActivityThis,ageFid);
jstring name = (jstring)env->GetObjectField(thiz,nameFid);//此处有编码转换问题未解决//修改属性值,C++中修改变量值后,Java重新获取打印发现是修改过的
env->SetIntField(mainActivityThis, ageFid , 11);
env->SetObjectField(thiz, nameFid,Stringvalue);//获取方法的methodId,--》这里就用到了签名类型
jmethodID addMid=env->GetMethodID(mainActivityCls, "add", "(II)I");
int result=env->CallIntMethod(mainActivityThis, addMid, 1, 1); //这里就能获取到2的值。

仔细看一下上面的代码,就大致能理解这个签名表格的具体作用:为了找到Java方法的参数和返回值的形式。

Java签名类型小结:

(1)基础类型签名那些转换都是很容易记住的,基础类型中,特别留意一下boolean类型 是 Z 就行
(2)对象Object类型的转换是:L+全包名(包名直接用 /间隔)+类名+分号
(3)数组类型签名转换:[数组类型,比如[I,表示Java的 int[](4)方法签名的转换:(参数类型)返回类型,中间多个参数类型依此填写就行,
比如:Jni中的代码:env->GetMethodID("add", "(IILjava/lang/String;Ljava/lang/String;)Ljava/lang/String;")
如果不清楚上面的表格转换,看起来就头大,特别是那些有三四个以上参数的情况,但是学习过后就不难了,
查看表格对应关系可以知道,Java中的对应方法是:public String add(int a,int b,String c,String d)
其实就是先看括号后面的返回值,然后再一个个确定括号内的形参变量

共勉: 这短短的一生,我们最终都会过去,你不妨大胆一些,爱一个人、攀一座山、追一个梦

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【C++杂货铺】详解string

目录 &#x1f308;前言&#x1f308; &#x1f4c1; 为什么学习string &#x1f4c1; 认识string&#xff08;了解&#xff09; &#x1f4c1; string的常用接口 &#x1f4c2; 构造函数 &#x1f4c2; string类对象的容量操作 &#x1f4c2; string类对象的访问以及遍历操…

js判断页面是否是在iframe里面

文章目录 一、前言1.1、_blank跳转1.2、_self跳转 二、方法2.1、判断子项目是否是在iframe内部2.2、实现_self跳转 三、最后 一、前言 上面是父前端项目里的iframe加载了子前端项目的页面。此时如果点击子项目的内容&#xff0c;如果要进行父项目浏览器页面跳转&#xff0c;可以…

C++初阶 类(上)

目录 1. 什么是类 2. 如何定义出一个类 3. 类的访问限定符 4. 类的作用域 5. 类的实例化 6. 类的大小 7. this指针 1.this指针的引出 2. this指针的特性 8. 面试题 1. 什么是类 在C语言中&#xff0c;不同类型的数据集合体是结构体。为了方便管理结构体&#xff0c;我…

HarmonyOS NEXT应用开发案例——自定义TabBar

介绍 本示例主要介绍了TabBar中间页面如何实现有一圈圆弧外轮廓以及TabBar页签被点击之后会改变图标显示&#xff0c;并有一小段动画效果。 效果图预览 使用说明&#xff1a; 依次点击tabBar页面&#xff0c;除了社区图标之外&#xff0c;其它图标往上移动一小段距离。 实现…

中霖教育:消防工程师报考条件汇总

消防工程师考试报名条件汇总&#xff0c;想要参加考试的考生可以根据自己的专业和学历判断自己是否符合条件。 1、大专学历&#xff0c;消防工程专业&#xff0c;工作年限满6年&#xff0c;从事消防安全技术工作满4年;消防工程相关专业&#xff0c;工作年限满7年&#xff0c;从…

实习记录课程内容

2024年3月1日18:21:09 第二周周五——方法篇 等量关系的构造 初中数学中&#xff0c;让学生构造等量关系的数学原理主要包括以下几个方面&#xff1a; 等式的性质&#xff1a;等式的两边同时加上或减去同一个数&#xff0c;等式仍然成立&#xff1b;等式的两边同时乘以或除以…

利用API接口进行竞品价格监控

在电子商务和零售行业&#xff0c;了解竞争对手的定价策略对于保持市场竞争力至关重要。随着技术的发展&#xff0c;通过编程接口&#xff08;API&#xff09;获取商品详情成为企业监控竞品价格的有效手段。本文将详细介绍如何利用API接口实现竞品价格监控的流程和策略。 第一步…

EventSource数据一次性出来

基于txt/event-stream的EventSource流&#xff0c;使用代理时需注意&#xff1a; 开发阶段如果使用vue自带的代理&#xff0c;需要关闭compress&#xff0c;即 devServer: {proxy: {/xx: {target: ip,changeOrigin: true,pathRewrite: {^/xx: /xx}},},compress: false}再nginx部…

浅谈字典攻击

一、前言 字典攻击是一种常见的密码破解方法&#xff0c;它使用预先编制的字典文件作为攻击字典&#xff0c;通过尝试猜测密码的方式来破解密码。下面是一个关于字典攻击的博客&#xff0c;希望能够为您了解字典攻击提供帮助。 二、字典攻击概述 字典攻击是一种密码破解方法&…

文献学习-14-一种用于高精度微创手术的纤维机器人

Authors: Mohamed E. M. K. Abdelaziz1,2 †, Jinshi Zhao1,3 †, Bruno Gil Rosa1,2 , Hyun-Taek Lee4 , Daniel Simon3,5 , Khushi Vyas1,2 , Bing Li6,7 , Hanifa Koguna3 , Yue Li1 , Ali Anil Demircali3 , Huseyin Uvet8 , Gulsum Gencoglan9,10, Arzu Akcay11,12, Moham…

CDR(CorelDRAW)2024最新汉化注册补丁包下载

CorelDRAW 2024是一款功能强大的平面设计软件&#xff0c;广泛应用于图形设计、编辑照片以及创建网站等领域。凭借对高级操作系统的支持、多监视器查看和4K显示屏的兼容性&#xff0c;它让初始用户、图形专家、小型企业主和设计爱好者都能自信快速地交付专业级结果。 CorelDRA…

一文教你搞懂Vue生命周期

Vue生命周期 生命周期示意图 Vue3 组件创建阶段 new vue new一个vue的实例对象&#xff1b;此时会进入组件的创建过程&#xff08;该组件在代码中被注册并使用时&#xff0c;就代表着其被new了一个新的实例对象&#xff09;。 Init Events & Lifecycle 初始化组件的事件和…

spring aop中获取request和response

Spring AOP 操作中如何使用request和response 实际使用时&#xff0c;如果方法一不行&#xff0c;请使用方法二 方法一 HttpServletRequest request ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getRequest(); HttpServletResponse respons…

Python执行 nohup 导致僵尸进程问题. /usr/bin/sh -> /usr/bin/bash

将 /usr/bin/sh -> dash 改成 /usr/bin/sh -> /usr/bin/bash 即可解决. ln -sf /usr/bin/bash /usr/bin/sh cmd" nohup python3 xxxx.py > /xx/xxx.log 2>&1 &" #在python里执行上边的命令ret subprocess.Popen(cmd,shellTrue,stdouts…

uniapp封装文字提示气泡框toolTip组件

uniapp封装文字提示气泡框toolTip组件 文字提示气泡框&#xff1a;toolTip 因为uniapp 中小程序中没有window对象&#xff0c;需手动调用 关闭 第一种办法关闭&#xff1a;this.$refs.tooltip.close() 第二种办法关闭&#xff1a;visible.sync false 移动端没有现成的toolTip组…

【2024.03.05】定时执行专家 V7.1 发布 - TimingExecutor V7.1 Release

目录 ▉ 软件介绍 ▉ 新版本 V7.1 下载地址 ▉ V7.1 新功能 ▼2024-03-03 V7.1 - 更新日志 ▉ V7.0 新UI设计 ▉ 软件介绍 《定时执行专家》是一款制作精良、功能强大、毫秒精度、专业级的定时任务执行软件。软件具有 25 种【任务类型】、12 种【触发器】触发方式&#x…

InstantiationAwareBeanPostProcessor学习

简介 InstantiationAwareBeanPostProcessor 是 Spring 框架中的一个核心接口&#xff0c;它允许在 Spring 容器实例化 bean 之前和之后执行自定义逻辑。这个接口扩展了 BeanPostProcessor 接口&#xff0c;并增加了与 bean 实例化和属性填充相关的回调方法。 这个接口定义的主…

swift 闭包捕获列表

以下函数会打印出什么&#xff1f; var car "Benz" let closure { [car] in print("I drive \(car)") } car "Tesla" closure() 因为 clousre 已经申明将 car 复制进去了&#xff08;[car]&#xff09;&#xff0c;此时clousre 里的 car…

Linux运维工程师不可或缺的10款工具

运维工程师在日常工作中频繁运用的10款工具&#xff0c;并细致阐述每款工具的功能、适用场景以及其卓越之处。 1. Shell脚本&#xff1a; 功能&#xff1a;主要用于自动化任务和批处理作业。 适用场景&#xff1a;频繁用于文件处理、系统管理、简单的网络管理等操作。 优势&…

[论文笔记] Transformer-XL

这篇论文提出的 Transformer-XL 主要是针对 Transformer 在解决 长依赖问题中受到固定长度上下文的限制,如 Bert 采用的 Transformer 最大上下文为 512(其中是因为计算资源的限制,不是因为位置编码,因为使用的是绝对位置编码正余弦编码)。 Transformer-XL 能学习超过固定长…