Java代码与Jni层之间传递数组(byte[])

在Java代码与Jni层之间传递数组(byte[]),一个典型的应用是Java层把需要发送给客户端的数据流传递到Jni层,由Jni层的Socket代码发送出去,当然,Jni层也需要把从Socket接收到的数据流返回给Java层。我简单地总结了一下,从Java层到Jni层,从Jni层到JAVA层,各有3种传递方式,下面用代码示例简单地介绍一下。

示例代码的主要文件有两个,一个是Native.java,是Java层的类;另一个是Native.c,是JNI层的文件,关键的地方我都用注释添加到代码中了,完整的代码见博文后面的附件。

一、 从Java传递数组到Jni层

Jni层接收到Java层传递过来的byte[]数组,一般有2个函数来获取它的值,一个 GetByteArrayRegion,另一个是 GetByteArrayElements ,前者是进行值拷贝,将Java端数组的数据拷贝到本地的数组中,后者是指针的形式,将本地的数组指针直接指向Java端的数组地址,其实本质上是JVM在堆上分配的这个数组对象上增加一个引用计数,保证垃圾回收的时候不要释放,从而交给本地的指针使用,使用完毕后指针一定要记得通过ReleaseByteArrayElements进行释放,否则会产生内存泄露。

首先看Native.java的定义:
在这里插入图片描述

再看看对应的native.c的实现代码:
在这里插入图片描述

二、 从Jni层传递数组到Java层

把Jni层定义的数组传递到Java层,一般有两种方法,一种是通过native函数的返回值来传递,另一种是通过jni层回调java层的函数来传递,后者多用于jni的线程中。无论哪种方法,都离不开 SetByteArrayRegion 函数,该函数将本地的数组数据拷贝到了 Java 端的数组中。下面只介绍前一种方式,即通过native函数返回值的方式传递jni层的数组,回调的方式其实用法类似,就不详细介绍了。

首先看Native.java的定义:
在这里插入图片描述

再看看native.c是如何实现的:
在这里插入图片描述

由上述代码示例可以看出,首先通过 NewByteArray 在堆上分配数组对象,然后通过SetByteArrayRegion 把本地的数组数据拷贝到堆上分配的数组中去,然后通过返回值将分配的数组对象返回到Java层即可。对于回调的方式,这几步操作也是一样的,唯一的不同是,回调方式不是以返回值的方式将数组对象返回给Java层,而是在回调函数中,以回调函数参数的形式返回给Java层。

三、 Direct Buffer 方式传递

Java和Jni层的数组传递还有一个比较重要的方式,就是通过Direct Buffer来传递,这种方式类似于在堆上创建创建了一个Java和Jni层共享的整块内存区域,无论是Java层或者Jni层均可访问这块内存,并且Java端与Jni端同步变化,由于是采用的是共享内存的方式,因此相比于普通的数组传递,效率更高,但是由于构造/析构/维护这块共享内存的代价比较大,所以小数据量的数组建议还是采用上述方式,Direct Buffer方式更适合长期使用频繁访问的大块内存的共享。具体使用方法介绍如下:

首先看Native.java的定义:
在这里插入图片描述

再看看native.c是如何实现的:
在这里插入图片描述

由上述代码可以看出,其中使用起来还是很简单的,Jni层只需要通过GetDirectBufferAddress函数即可获取到这块共享的内存的地址,Direct Buffer的管理工作均由操作系统来负责。

  • 8
    点赞
  • 28
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要将JNI中的char*类型转换为Java中的byte数组,可以按照以下步骤进行操作: 1. 获取char*类型字符串的长度,使用strlen()函数可以获取。 2. 创建一个jbyteArray类型的Java数组,长度为字符串长度,使用NewByteArray()函数可以创建。 3. 将char*类型字符串转换为jbyte类型数组,使用jbyte*类型的指针指向char*类型字符串,然后使用SetByteArrayRegion()函数将每个char类型元素转换为jbyte类型并设置到Java数组中。 4. 释放char*类型字符串的内存空间,使用free()函数释放。 下面是一个示例代码: ```c++ JNIEXPORT jbyteArray JNICALL Java_com_example_MyClass_charToByteArray(JNIEnv *env, jobject obj, jcharArray charArray) { // 将jcharArray类型转换为jchar类型指针 jchar *chars = env->GetCharArrayElements(charArray, NULL); // 获取字符串长度 int len = strlen(chars); // 创建jbyteArray类型的Java数组 jbyteArray byteArray = env->NewByteArray(len); // 将char*类型字符串转换为jbyte类型的数组 jbyte *bytes = (jbyte*)chars; env->SetByteArrayRegion(byteArray, 0, len, bytes); // 释放char*类型字符串的内存空间 free(chars); // 返回jbyteArray类型的Java数组 return byteArray; } ``` 在上面的示例代码中,我们首先将输入的jcharArray类型转换为jchar类型指针,然后获取字符串长度。接着,我们使用NewByteArray()函数创建一个长度为字符串长度的jbyteArray类型的Java数组。然后,我们将jchar类型指针转换为jbyte类型指针,并使用SetByteArrayRegion()函数将每个char类型元素转换为jbyte类型并设置到Java数组中。最后,我们使用free()函数释放char*类型字符串的内存空间,并返回jbyteArray类型的Java数组

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值