当前位置: 首页 > news >正文

Android JNI浅析、Java和Native通信对象的传值和回调

简单了解一下jni

JNI是一个本地编程接口,它允许运行在Java虚拟机的Java代码与用其他语言(如C,C++和汇编)编写的库交互。

jni函数签名

首先看一下java类型对应的jni类型:

Java类型符号
BooleanZ
ByteB
CharC
ShortS
IntI
LongJ
FloatF
DoubleD
VoidV
数组[ 比如:int[] -> [I ,如果是二维数组 int[][] -> [[I
objects以"L"开头,以";"结尾,中间是用"/" 隔开的包及类名。比如:Ljava/lang/String;如果是嵌套类,则用$来表示嵌套。

比如:

privite native int test(String arg);

则它的签名为:

(Ljava/lang/String;)I

函数参数的传递

  • 基本类型(如整型,字符等)在Java和native之间是采用值传递
  • Java对象采用的是引用传递

关于局部引用的相关内容,可以参考之前的文章:
JNI内存方面说明以及相关类型手动释放内存

JavaVM和JNIEnv

这两个结构体在jni.h中有定义:

#if defined(__cplusplus)
typedef _JNIEnv JNIEnv;
typedef _JavaVM JavaVM;
#else
typedef const struct JNINativeInterface* JNIEnv;
typedef const struct JNIInvokeInterface* JavaVM;
#endif

这里也能看出c版本和c++版本,在使用调用上有一些不同,c++相当于又包了一层!

下边看一下c++的版本:
JavaVM

struct _JavaVM {const struct JNIInvokeInterface* functions;#if defined(__cplusplus)jint DestroyJavaVM(){ return functions->DestroyJavaVM(this); }jint AttachCurrentThread(JNIEnv** p_env, void* thr_args){ return functions->AttachCurrentThread(this, p_env, thr_args); }jint DetachCurrentThread(){ return functions->DetachCurrentThread(this); }jint GetEnv(void** env, jint version){ return functions->GetEnv(this, env, version); }jint AttachCurrentThreadAsDaemon(JNIEnv** p_env, void* thr_args){ return functions->AttachCurrentThreadAsDaemon(this, p_env, thr_args); }
#endif /*__cplusplus*/
};

JNIEnv

struct _JNIEnv {/* do not rename this; it does not seem to be entirely opaque */const struct JNINativeInterface* functions;#if defined(__cplusplus)jint GetVersion(){ return functions->GetVersion(this); }jclass DefineClass(const char *name, jobject loader, const jbyte* buf,jsize bufLen){ return functions->DefineClass(this, name, loader, buf, bufLen); }jclass FindClass(const char* name){ return functions->FindClass(this, name); }......
}    

总的来说:

JavaVM:是java虚拟机环境,每个进程有且只有只有一个。
JNIEnv: 是线程上下文环境,每个线程只有一个,不能跨线程。

我们可以通过JNIEnv来获取一个JavaVM:

jint GetJavaVM(JNIEnv *env, JavaVM **vm);// vm:用来存放获得的虚拟机的指针的指针。
// return:成功返回0,失败返回其他。

也可以通过JavaVM来获取一个JNIEnv:

jint GetEnv(void** env, jint version)

上边也提到了JNIEnv是线程绑定的,所以通常,用全局的JavaVM获取一个当前线程的JNIEnv的时候,通常需要绑定到当前线程:

char thread_name[128] = { 0 };
prctl(PR_GET_NAME, (char *)(thread_name));
JavaVMAttachArgs args;
args.version = JNI_VERSION_1_6;
args.name = thread_name;
args.group = NULL;
gJvm->AttachCurrentThread(&pEnv, (void *)(&args))

同样的,也需要在不使用的时候解绑:

gJvm->DetachCurrentThread();

本地函数的静态加载和动态加载

所谓静态加载,就是我们常见的,jni函数的声明是Java_包名_类名_方法名(参数...)这样子的,Java方法和本地函数之间的映射关系编译器已经帮我们做了。

下边了解一下动态加载:

JNIEXPORT jint JNI_OnLoad(JavaVM* vm, void* reserved);
JNIEXPORT void JNI_OnUnload(JavaVM* vm, void* reserved);

JNI_OnLoad方法是在动态库被加载时调用,而JNI_OnUnload则是在本地库被卸载时调用。所以这两个函数就是一个本地库最重要的两个生命周期方法。

在JNI_OnLoad()中,就可以手动注册本地函数,做好对java方法的映射。
一般的可以这个样子:

const JNINativeMethod gMethods[] = {{"native函数名", "native函数签名", (void *)native函数}, //第三个为函数指针...//别的native函数
};JNIEXPORT jint JNI_OnLoad(JavaVM* vm, void* reserved){JNIEnv *env = NULL;jint result = -1;if (vm->GetEnv((void **) &env, JNI_VERSION_1_6) != JNI_OK) {return result;}// 动态注册native functionjclass clazz = env->FindClass("com.xx.xx.xxclass");    env->RegisterNatives(clazz, &method, 1) //第三个参数为总的数量// 返回jni的版本return JNI_VERSION_1_6;
}

访问字段和函数

对于jni函数中传入的jobject对象,要想访问对象的字段和函数,就需要先获取对应的class引用!

jobject testObject;   //假设我们已经有了这么一个对象jclass testCls = env->GetObjectClass(testObject); //获取class引用
//int testFiled;
//int test(String arg); 
jfieldID testFiledId = env->GetFieldID(testCls, "testFiledID", "I");
jmethodID testMethodId = env->GetMethodID(testCls, "test", "(Ljava/lang/String;)I");// 注意对字段的get/set,函数的调用,都必须使用具体的jobject对象,而不能是class引用
env->GetIntField(testObjcet, testFiledId);
env->CallIntMethod(testObject, testMethodId, "arg...");

java像Native中的对象的传递,一般也是安装上边的方式来进行,取出字段的值,再赋值给native对应的对象!

如果是native的对象回调到java层怎么做?

// 先找到class引用
jclass jcs = env->FindClass("com/xx/xx/xxclass");
// 创建对象有两种方法
// 第一种是调用类的构造函数:mthodId就是构造函数的id
jmethodID cls_constructor = env->GetMethodID(jcs, "<init>", "()V");jobject NewObject(JNIEnv *env, jclass clazz, jmethodID methodID, ...);// 第二种是直接alloc对象
jobject AllocObject(jclass clazz)//然后将native对象赋值给jobject即可!
http://www.lryc.cn/news/39794.html

相关文章:

  • linux目录/usr/lib/systemd/system目录详解
  • 408考研计算机之计算机组成与设计——知识点及其做题经验篇目4:CPU的功能和基本结构
  • 2022-12-10青少年软件编程(C语言)等级考试试卷(五级)解析
  • 刷题专练之链表(一)
  • elasticsearch高级查询api
  • 力扣-股票的资本损益
  • 蓝桥杯刷题冲刺 | 倒计时26天
  • 嵌入式软件开发之Linux 用户权限管理
  • 2023-03-15 RabbitMQ
  • 二叉树链式结构的实现
  • 蓝桥杯刷题冲刺 | 倒计时28天
  • 一文带你吃透操作系统
  • 计算机网络英文简称汇总
  • 腾讯云云服务器标准型S5性能配置简单测评
  • RK3568平台开发系列讲解(Linux系统篇)消息队列
  • 2021电赛国一智能送药小车(F题)设计报告
  • 刚工作3天就被裁了....
  • docker安装elasticsearch与head教程完整版—.NET Core Web Api与elasticsearch打造全站全文搜索引擎
  • 蓝桥冲刺31天之315
  • 常见排序算法
  • C语言实现学生成绩管理系统思考
  • C++11中Lambda新特性
  • 【jvm系列-01】初识虚拟机与java虚拟机
  • 「Python 基础」数据库应用编程
  • 一个nginx的小项目,不写代码,实现在局域网内访问其他电脑的网页
  • 23.3.14打卡 2022年江西省大学生程序设计竞赛(正式赛)ABL
  • 用idea操作hbase数据库,并映射到hive
  • 手机解锁方法:8个顶级的 Android 手机解锁软件
  • JVS快速开发平台2.1.7版本,列表页配置新增特性介绍
  • 【华为机试真题详解 Python实现】去除多余空格【2023 Q1 | 100分】