JNI内两种方式从C/C++中传递一维、二维、三维数组数据至Java层详细梳理

news2024/9/21 10:45:58

目录

0 前言

1 准备工作介绍

2 一维数组

2.1 return形式

2.2 参数形式

3 二维数组

3.1 return形式

3.2 参数形式

4 三维数组

4.1 return形式

4.2 参数形式

5 测试代码

6 结果说明


0 前言

        就如之前我写过的一篇文章【JNI内形参从C代码中获取返回值并返回到Java层使用】中所描述的一样,JNI编程往往需要考虑的就是如何将C/C++中计算分析得到的数据传递至Java层。传递的方式就有两种:①直接通过函数return出来;②再就是通过形参获取得到。在以上文章内介绍了int值得传递情况,这篇文章则具体讲述数组得传递方式和操作,包括一维数组、二维数组以及三维数组,数组使用到三维基本满足了绝大部分的使用需求了。

        之所以写这篇文章,也是因为检索了很久对于数组在JNI内传递数据方式只看到第一种通过return方式传递的。为此就将自己工作整理出来,实现数组通过形参的方式传递数据给java层。

1 准备工作介绍

        在写测试案例之前需要有系列的准备工作,由于这不是重点,该部分我简单介绍一下。首先需要创建java工程,然后编写java文件,生成JNI头文件,最后编译成动态库。这里面具体可以参考:【IntelliJ IDEA平台下JNI编程(一)—HelloWorld篇_idea jni_走召大爷的博客-CSDN博客】

        本次编写环境Ubuntu18.04 、gcc7.5.0、生成.so动态库、IDEA平台。以下分别介绍在jni层不同维度数组不同形式传递数据的代码写法,具体介绍可以看代码内部注释。

2 一维数组

2.1 return形式

//一维数组  return形式向JAVA传递数据
JNIEXPORT jdoubleArray Java_com_test_java_JNItest_test0
  (JNIEnv *env, jobject)
{
	int n = 3;	
	//构造/模拟一个数组
	double test[3] = {0.0,1.0,2.0};
	
	//定义一维数组,数组元素个数为n=3
    jdoubleArray one = env->NewDoubleArray(n);

	env->SetDoubleArrayRegion(one,0,n,test);	
	return one;

}

2.2 参数形式

//一维数组  形参向JAVA传递数据
JNIEXPORT jint Java_com_test_java_JNItest_test3
(JNIEnv *env, jobject,jdoubleArray result)
{
	int n = 3;
	//构造/模拟一个数组
	double test[3] = {0.0,1.0,2.0};
	
	env->SetDoubleArrayRegion(result,0,n,test);	
	return 0;	
}

3 二维数组

3.1 return形式

//二维数组  return形式向JAVA传递数据
JNIEXPORT jobjectArray Java_com_test_java_JNItest_test1
  (JNIEnv *env, jobject)
{
	//构造二维数组
	double test[2][3] = {{1.0,2.0,3.0},{3.0,3.0,6.0}};
	
        //一维数组  类型定义
        jclass ArrCls1 = env->FindClass("[D");  
        //创建一个有polygonpointNum个元素,每个元素的值是NULL的一维数组的数组  
        jobjectArray second = env->NewObjectArray(2, ArrCls1, NULL);  //等同二维数组
	//对于二维数组其内容是每一个一维数组,那么就循环定义一维数组,直接存入结果里面  
	//对于测试的二维数组:其为2x3固定的,不然通过下面循环可以获取变长的数组内容
        for(int i = 0;i < 2;i++)
        {	
        	int num = 3;
		//定义一维数组,数组元素个数为num=3
        	jdoubleArray darr = env->NewDoubleArray(num);
        	//将test的内容赋值给darr
        	env->SetDoubleArrayRegion(darr, 0, num, test[i]);
        	//然后再将darr一维数组内容直接按顺序赋值给second二维数组
        	env->SetObjectArrayElement(second, i, darr);  
      		//删除临时元素darr数组  
    		env->DeleteLocalRef(darr); 
        }		
	return second;


}

3.2 参数形式

//二维数组  形参向JAVA传递数据
JNIEXPORT jint Java_com_test_java_JNItest_test4
  (JNIEnv *env, jobject, jobjectArray  result)
{
	//构造二维数组
	double test[2][3] = {{1.0,2.0,3.0},{3.0,3.0,6.0}};
	//对于二维数组其内容是每一个一维数组,那么就循环定义一维数组,直接存入结果里面  
	//对于测试的二维数组:其为2x3固定的,不然通过下面循环可以获取变长的数组内容
        for(int i = 0;i < 2;i++)
        {	
        	int num = 3;
		//定义一维数组,数组元素个数为num=3
        	jdoubleArray darr = env->NewDoubleArray(num);
        	//将test的内容赋值给darr
        	env->SetDoubleArrayRegion(darr, 0, num, test[i]);
        	//然后再将darr一维数组内容直接按顺序赋值给二维结果数组,如此就将c/c++内部的数组内容通过result传递到java层了
        	env->SetObjectArrayElement(result, i, darr);  
      		//删除临时元素darr数组  
    		env->DeleteLocalRef(darr); 
        }		
	return 0;

}

4 三维数组

4.1 return形式

//三维数组  return形式向JAVA传递数据
JNIEXPORT jobjectArray Java_com_test_java_JNItest_test2
  (JNIEnv *env, jobject)
{
	//构造一个三维数组,示例为两个多边形,每个多边形3个顶点,每个顶点由x,y组成
	double test[2][3][2] = {{{1.0,2.0},{3.0,5.0},{2.0,4.0}},{{1.0,2.0},{3.0,5.0},{2.0,4.0}}};
  	int polygonNum = 2;
  	
  	//二维数组类型定义
  	jclass ArrCls2 = env->FindClass("[[D");          
    	
    	//实例化三维数组对象,第一个参数数据的大小,第二个参数用来实例化用的类是一个二维数组,也就是数组里的每个元素都是一个二维数组    	  
  	jobjectArray three = env->NewObjectArray(polygonNum, ArrCls2, NULL);  //这样three就是一个三维数组。
  	for(int i = 0;i < polygonNum;i++)  
  	{ 
  	       int polygonpointNum =  3;
   	        //一维数组  类型定义
   	        jclass ArrCls1 = env->FindClass("[D");  
  	        //创建一个有polygonpointNum个元素,每个元素的值是NULL的一维数组的数组  
   	        jobjectArray second = env->NewObjectArray(polygonpointNum, ArrCls1, NULL);  //等同二维数组
   	          
  	        for (int j = 0; j < polygonpointNum; j++) 
  	        {  
  	        	jdouble tmp[2]; //构造每一个一维存储单元/* make sure it is large enough! */
    	            	tmp[0] = test[i][j][0];  
     	     	  	tmp[1] = test[i][j][1];    //也可以不构造和赋值,后面one的赋值可以env->SetDoubleArrayRegion(one, 0, 2, test[i][j]);也可以的 
     	     	  	
  	        	//定义有两个元素的一维数组
     	      	 	jdoubleArray one = env->NewDoubleArray(2);  
      	    	  
    	        	//把tmp里数据从0开始传递2个到one里 
     	       		env->SetDoubleArrayRegion(one, 0, 2, tmp);  
    	        	//给二维数组second的第j个元素设置值  
      	      		env->SetObjectArrayElement(second, j, one);  //注意理解内容填充(数组初始化)
      	      		//删除临时元素one数组  
      	      		env->DeleteLocalRef(one);  
      	 	}  
            	//给三维数组里的第i个元素设置值,值是一个有一个元素组成的二维数据
            	env->SetObjectArrayElement(three, i, second);  
            	//删除临时元素二维数组  
            	env->DeleteLocalRef(second);  
    	} 
    	
    	return three;
}

4.2 参数形式

//三维数组  形参向JAVA传递数据
JNIEXPORT jint Java_com_test_java_JNItest_test5
  (JNIEnv *env, jobject, jobjectArray  result)
{
	//构造一个三维数组,示例为两个多边形,每个多边形3个顶点,每个顶点由x,y组成
	double test[2][3][2] = {{{1.0,2.0},{3.0,5.0},{2.0,4.0}},{{1.0,2.0},{3.0,5.0},{2.0,4.0}}};
	
	int polygonNum = 2;
  	for(int i = 0;i < polygonNum;i++)  
  	{ 
  	       int polygonpointNum =  3;
   	        //一维数组  类型
   	        jclass arrClass = env->FindClass("[D");  
  	        //创建一个有polygonpointNum个元素,每个元素的值是NULL的一维数组的数组  
   	        jobjectArray second = env->NewObjectArray(polygonpointNum, arrClass, NULL);  //等同二维数组
   	        //给一维数据填充值  
  	        for (int j = 0; j < polygonpointNum; j++) 
  	        {  
    	            	jdouble tmp[2]; //构造每一个一维存储单元
    	            	tmp[0] = test[i][j][0];  
     	     	  	tmp[1] = test[i][j][1]; //也可以不构造和赋值,后面one的赋值可以env->SetDoubleArrayRegion(one, 0, 2, test[i][j]);也可以的
     	     	  	
     	     	  	//创建一个有2个元素的一维数组
     	      	 	jdoubleArray one = env->NewDoubleArray(2);  
      	    	  	   
    	        	//把tmp里数据从0开始传递2个到one里  
     	       		env->SetDoubleArrayRegion(one, 0, 2, tmp);  
    	        	//给二维数组second的第j个元素设置值  
      	      		env->SetObjectArrayElement(second, j, one);  //注意理解内容填充(数组初始化)
      	      		//删除临时元素one数组  
      	      		env->DeleteLocalRef(one);  
      	 	}  
            	//给需要传出的三维数组里的每个元素设置值,值是一个有一个元素组成的二维数据
            	env->SetObjectArrayElement(result, i, second);  
            	//删除临时元素二维数组  
            	env->DeleteLocalRef(second);  
    	} 
    	return 0;

}

5 测试代码

        在java内的测试代码如下:

package com.test.java;

/**
 * @author yh
 * @version 1.0
 * @date 23-3-6 下午8:58
 */
public class JNItest {
    static {
        System.load("/root/workspace-yh/javaProjectTest/jni/libjnitest.so");
    }

    public native double[] test0();
    public native double[][] test1();
    public native double[][][] test2();

    public native int test3(double[] result);
    public native int test4(double[][] result);
    public native int test5(double[][][] result);

    public void printOneArray(double[] t){
        int n = t.length;
        for(int i = 0;i < n;i++){
            System.out.println(t[i]);
        }
    }

    public void printSecondArray(double[][] t){
        int n = t.length;
        for(int i = 0;i < n;i++){
            int m = t[i].length;
            String temp = "";
            for(int j = 0;j < m;j++){
                temp += t[i][j]+",";
            }
            System.out.println(temp);
        }
    }

    public void printThreeArray(double[][][] t){
        int n = t.length;
        for(int i = 0;i < n;i++){
            int m = t[i].length;
            for(int j = 0;j < m;j++){
                String temp = "";
                int k = t[i][j].length;
                for(int l = 0;l < k;l++){
                    temp += t[i][j][l]+",";
                }
                System.out.println(temp);
            }
        }
    }

    public  static void main(String[] args){
        JNItest JNI = new JNItest();
        double[] a3 = new double[3];
        double[][] a4 = new double[2][2];
        double[][][] a5 = new double[2][3][2];

        System.out.println("-------------------test0-----------------");
        double[] a0 = JNI.test0();
        JNI.printOneArray(a0);
        System.out.println("-------------------test3-----------------");
        int rnt3 = JNI.test3(a3);
        JNI.printOneArray(a3);

        System.out.println("-------------------test1-----------------");
        double[][]a1 = JNI.test1();
        JNI.printSecondArray(a1);
        System.out.println("-------------------test4-----------------");
        int rnt4 = JNI.test4(a4);
        JNI.printSecondArray(a4);

        System.out.println("-------------------test2-----------------");
        double[][][]a2 = JNI.test2();
        JNI.printThreeArray(a2);
        System.out.println("-------------------test5-----------------");
        int rnt5 = JNI.test5(a5);
        JNI.printThreeArray(a5);
    }    

}

6 结果说明

        直接执行main函数,控制台打印信息:

         根据结果可以看到,两种方式均把c/c++层数据传递到了java层了。那么后面根据工程需要想怎么传递数据就怎么传递数据,想怎么使用接口就怎么设计接口了。

参考文章:

JNI 返回二维、三维,char、float、int、long型数组到java层_谢文浩博客-CSDN博客_jni 返回二维数组

Android Studio开发之JNI层开发 --- jni层返回二维数组对象_Jimmy-CSDN博客_jni 返回二维数组

android jni jobjectArray存储输出不同类型的数据_阿文的博客-CSDN博客_jni jobjectarray

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

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

相关文章

Elsevier上传LaTeX 修改稿踩坑

背景 千辛万苦修改完论文&#xff0c;结果发现要求上传可编辑文件&#xff0c;tex上传真的太难了&#xff0c;一堆坑&#xff0c;尤其是编译错误&#xff0c;要等系统创建pdf后才能找到。中间还打了北京的客服电话&#xff0c;结果他们那边并不懂相关的东西。说latex是第三方公…

C语言-基础了解-22-C文件读写

C文件读写 一、打开文件 可以使用 fopen( ) 函数来创建一个新的文件或者打开一个已有的文件&#xff0c;这个调用会初始化类型 FILE 的一个对象&#xff0c;类型 FILE 包含了所有用来控制流的必要的信息。下面是这个函数调用的原型&#xff1a; FILE *fopen( const char *fil…

JS中undefined和null的区别

● JavaScript 真是一个特殊的语言, 其他语言都只有一个表示 “无” 的值, 比如 Java 语言用的是 null, C 语言用的是 NULL, Python 语言用的是 None, Ruby 语言用的是 nil. 只有 JS 里面表示 “空” 的有两个, 一个是 undefined, 一个是 null ● 很多小朋友在刚开始学习的时候…

【项目设计】高并发内存池(六)[细节优化+测试]

&#x1f387;C学习历程&#xff1a;入门 博客主页&#xff1a;一起去看日落吗持续分享博主的C学习历程博主的能力有限&#xff0c;出现错误希望大家不吝赐教分享给大家一句我很喜欢的话&#xff1a; 也许你现在做的事情&#xff0c;暂时看不到成果&#xff0c;但不要忘记&…

[ 网络 ] 应用层协议 —— HTTP协议

目录 1.HTTP协议 1.1URL urlencode和urldecode 2. HTTP协议格式 HTTP请求 HTTP响应 3.告知服务器意图的HTTP方法 GET&#xff1a;获取资源 POST&#xff1a;传输实体主体 GET和POST的区别 使用Cookie的状态管理 4.返回结果的HTTP状态码 状态码告知从服务器端返回的…

十连胜!实在智能×浙江省十家农商行,数字科技赋能普惠金融

近日&#xff0c;中共中央、国务院印发了《数字中国建设整体布局规划》&#xff08;以下简称《规划》&#xff09;。《规划》指出&#xff0c;建设数字中国是数字时代推进中国式现代化的重要引擎&#xff0c;是构筑国家竞争新优势的有力支撑。全面提升数字中国建设的整体性、系…

C语言-基础了解-23-C预处理器

C预处理器 一、C预处理器 C 预处理器不是编译器的组成部分&#xff0c;但是它是编译过程中一个单独的步骤。简言之&#xff0c;C 预处理器只不过是一个文本替换工具而已&#xff0c;它们会指示编译器在实际编译之前完成所需的预处理。我们将把 C 预处理器&#xff08;C Prepr…

spring-security2

参考b站up主&#xff1a;传送门 前沿 &#xff1a; 回顾上章&#xff1a;最顶层就是一个过滤器 SpringSecurity流程&#xff1a;FilterChainProxy->根据请求决定一个 SecurityFilterChain (挨个调用其matches方法 匹配请求能否处理)-> 执行SecurityFilterChain中的一系列…

Java简单认识泛型——图文详解

写在开头:想必大家和博主一样&#xff0c;在以往学习JavaSE的语法中&#xff0c;遇到了一个陌生的词——泛型&#xff0c;博主当时很好奇&#xff0c;什么是泛型呢&#xff1f;即使是学完了JavaSE&#xff0c;这个问题都没有解决&#xff0c;只能在百度查阅了解关于泛型的一些皮…

Maven依赖的基本概念

目录 1.依赖的基本配置 2.依赖范围 3.传递性依赖 1.依赖的基本配置 根元素project下的dependencies可以包含多个 dependence元素&#xff0c;以声明多个依赖。每个依赖都应该包含以下元素&#xff1a; 1. groupId, artifactId, version : 依赖的基本坐标&#xff0c; 对于任…

【C和C++】输出100内能够被13整除的数,取模判断方法

目录 前言基础概念重温整除例子小知识点收尾前言 在软件行业已经有快十年,技术虽然一般般,但是足够应付和解决编程入门的相关问题! 都说十年磨一剑,积累到一定经验,是时候发挥自己的价值,给予入门的同行些许的帮助! 为什么要写收费专栏,其实原因很简单,时间就是金钱(…

聚观早报 |拼多多跨境电商业务正式登陆澳洲;中国加快6G网络研发

今日要闻&#xff1a;拼多多跨境电商业务正式登陆澳洲&#xff1b;全球自动驾驶公司排名特斯拉垫底&#xff1b;中国将加快 6G 网络研发&#xff1b;B站再次“崩”上热搜&#xff01;已闪电修复&#xff1b;微软将必应AI聊天每次对话上限增加至8条拼多多跨境电商业务正式登陆澳…

大神教你在 Linux 中查看你的时区

在这篇短文中&#xff0c;我们将向你简单介绍几种 Linux 下查看系统时区的简单方法。在 Linux 机器中&#xff0c;尤其是生产服务器上的时间管理技能&#xff0c;是在系统管理中一个极其重要的方面。Linux 包含多种可用的时间管理工具&#xff0c;比如 date 或 timedatectlcomm…

自动化测试——执行javaScript脚本

文章目录一、点击元素(对应的click())二、input标签对应的值&#xff08;对应的send_keys()&#xff09;修改时间控件的属性值&#xff1a;三、元素的文本属性四、js脚本滚动操作一、点击元素(对应的click()) 使用场景&#xff1a;当使用显性等待不能解决问题时 代码中实现点击…

[神经网络]DETR目标检测网络

一、概述 相较于传统目标检测&#xff0c;DETR是一种纯端到端的网络。它不再需要NMS(非极大值抑制&#xff0c;用于去除多余的预测框)和生成anchor。 DETR提出了一个新的目标函数&#xff08;二分图匹配&#xff09;&#xff0c;这个函数可以强制网络输出一个独一无二的预测值&…

【Unity大气渲染】Unity Shader中实现大气散射(半成品)

写在前面 这是之前在做天空盒的时候同步写的分析博客&#xff0c;结果后面写到一半就忘了继续了&#xff0c;这里先贴出当时写的半成品&#xff0c;有小伙伴问我怎么做的&#xff0c;这里只能尽力把之前的半成品先放出来了&#xff08;写得很乱&#xff0c;勿怪orz&#xff09…

用C语言实现一个任意类型的队列

下面是一个简单的无类型队列的实现&#xff1a; #include <stdio.h> #include <stdlib.h>typedef struct Node {void *data;struct Node *next; } Node;typedef struct Queue {Node *front;Node *rear;int size; } Queue;void enqueue(Queue *queue, void *data);…

[蓝桥杯] 枚举、模拟和排列问题

文章目录 一、连号区间数 1、1 题目描述 1、2 题解关键思路与解答 二、递增三元组 2、1 题目描述 2、2 题解关键思路与解答 三、错误票据 3、1 题目描述 3、2 题解关键思路与解答 四、回文日期 4、1 题目描述 4、2 题解关键思路与解答 五、归并排序 标题&#xff1a;蓝桥杯——…

windows和linux出现timewait过多的解决方法

一、timewait出现在客户端还是服务端以及什么情况下出现 我是做性能测试的。在压测过程中遇到了timewait过多的情况&#xff0c;下面来看一下timewait产生的原因及解决办法&#xff0c;我自己在服务器起了一个很简单的springboot应用来验证自己的猜想及解决办法。 说到产生原…

基于docker部署prometheus

1、prometheus架构 Prometheus Server: 收集指标和存储时间序列数据&#xff0c;并提供查询接口 ClientLibrary:客户端库 Push Gateway: 短期存储指标数据。主要用于临时性的任务 Exporters:采集已有的第三方服务监控指标并暴露 metrics Alertmanager:告警 Web UI :简单的…