Java中JNI的使用详解第五篇:C/C++中操作Java中的数组
在Java中数组分为两种:
1.基本类型数组
2.对象类型(Object[])的数组(数组中存放的是指向Java对象中的引用)
一个能通用于两种不同类型数组的函数:
GetArrayLength(jarray array);
首先来看一下怎么处理基本类型的数组:
(1) Get<Type>ArrayElements(<Type>Array arr , jboolean* isCopide);
这类函数可以把Java基本类型的数组转换到C/C++中的数组,有两种处理方式,一种是拷贝一份传回本地代码,另一个是把指向Java数组的指针直接传回到本地代码中,处理完本地化的数组后,通过Release<Type>ArrayElements来释放数组
(2) Release<Type>ArrayElements(<Type>Array arr , <Type>* array , jint mode)
用这个函数可以选择将如何处理Java跟C++的数组,是提交,还是撤销等,内存释放还是不释放等
mode可以取下面的值:
0 :对Java的数组进行更新并释放C/C++的数组
JNI_COMMIT :对Java的数组进行更新但是不释放C/C++的数组
JNI_ABORT:对Java的数组不进行更新,释放C/C++的数组
(3) GetPrimittiveArrayCritical(jarray arr , jboolean* isCopied);
(4) ReleasePrimitiveArrayCritical(jarray arr , void* array , jint mode);
也是JDK1.2出来的,为了增加直接传回指向Java数组的指针而加入的函数,同样的也会有同GetStringCritical的死锁的问题
(5) Get<Type>ArrayRegion(<Type>Array arr , jsize start , jsize len , <Type>* buffer);
在C/C++预先开辟一段内存,然后把Java基本类型的数组拷贝到这段内存中,这个方法和之前拷贝字符串的GetStringRegion方法的原理是类似的
(6) Set<Type>ArrayRegion(<Type>Array arr , jsize start , jsize len , const <Type>* buffer);
把Java基本类型的数组中的指定范围的元素用C/C++的数组中的元素来赋值
(7) <Type>Array New<Type>Array(jsize sz)
指定一个长度然后返回相应的Java基本类型的数组
在来看一下怎么处理对象型数组
JNI没有提供直接把Java的对象类型数组(Object[])直接转到C++中的Object[]数组的函数,而是直接通过Get/SetObjectArrayElement这样的函数来对Java的Object[]数组进行操作由于去的对象数组没有进行拷贝,所以不需要释放任何资源
NewObjectArray可以通过指定长度跟初始值来创建某个类的数组
下面来看一下例子:操作两种类型的数组
Java中的代码:
- package com.jni.demo;
- public class JNIDemo {
- //定义一个int型数组
- int[] arrays = {4,3,12,56,1,23,45,67};
- //定义Father对象数组
- Father[] objArrays = {new Father(),new Father(),new Father()};
- //定义一个本地方法
- public native void callCppFunction();
- public static void main(String[] args)throws Exception{
- //调用动态链接库
- System.loadLibrary("JNIDemo");
- JNIDemo jniDemo = new JNIDemo();
- jniDemo.callCppFunction();
- }
- }
- </span>
C++中的代码:
- #include<iostream>
- #include"com_jni_demo_JNIDemo.h"
- #include<algorithm>
- using namespace std;
- JNIEXPORT void JNICALL Java_com_jni_demo_JNIDemo_callCppFunction (JNIEnv * env, jobject obj)
- {
- //获取Java中数组属性arrays的id
- jfieldID fid_arrays = env->GetFieldID(env->GetObjectClass(obj),"arrays","[I");
- //获取Java中数组属性arrays的对象
- jintArray jint_arr = (jintArray)env->GetObjectField(obj,fid_arrays);
- //获取arrays对象的指针
- jint* int_arr = env->GetIntArrayElements(jint_arr,NULL);
- //获取数组的长度
- jsize len = env->GetArrayLength(jint_arr);
- //打印数组中的值
- cout<<"数组的值为:";
- for(int s =0;s<len;s++){
- cout<<int_arr[s]<<',';
- }
- cout<<endl;
- //新建一个jintArray对象
- jintArray jint_arr_temp = env->NewIntArray(len);
- //获取jint_arr_temp对象的指针
- jint* int_arr_temp = env->GetIntArrayElements(jint_arr_temp,NULL);
- //计数
- jint count = 0;
- //偶数位存入到int_arr_temp内存中
- for(jsize j=0;j<len;j++){
- if(j%2==0){
- int_arr_temp[count++] = int_arr[j];
- }
- }
- //打印int_arr_temp内存中的数组
- cout<<"数组中位置是偶数的值为:";
- for(jsize k=0;k<count;k++){
- cout<<int_arr_temp[k]<<',';
- }
- cout<<endl;
- //将数组中一段(0-2)数据拷贝到内存中,并且打印出来
- jint* buffer = new jint[len];
- //获取数组中从0开始长度为3的一段数据值
- env->GetIntArrayRegion(jint_arr,0,3,buffer);
- cout<<"打印数组中0-3一段值:";
- for(int l=0;l<3;l++){
- cout<<buffer[l]<<',';
- }
- cout<<endl;
- //将数组中的一段(3-7)设置成一定的值,并且打印出来
- jint* buffers = new jint[4];
- for(int n=0;n<4;n++){
- buffers[n] = n+1;
- }
- //将buffers这个数组中值设置到数组从3开始长度是4的值中
- env->SetIntArrayRegion(jint_arr,3,4,buffers);
- //从新获取数组指针
- int_arr = env->GetIntArrayElements(jint_arr,NULL);
- cout<<"数组中3-7这段的值变成了:";
- for(int m=0;m<len;m++){
- cout<<int_arr[m]<<',';
- }
- cout<<endl;
- //调用C++标准库中的排序方法sort(...),传递一个数组的开始指针和结束指针
- std::sort(int_arr,int_arr+len);
- //迭代打印数组中的元素
- cout<<"数组排序后的结果:";
- for(jsize i=0;i<len;i++){
- cout<<int_arr[i]<<',';
- }
- cout<<endl;
- //释放数组指针
- env->ReleaseIntArrayElements(jint_arr,int_arr,JNI_ABORT);
- //获取Java中对象Father数组属性的id
- jfieldID fid_obj_arrays = env->GetFieldID(env->GetObjectClass(obj),"objArrays","[Lcom/jni/demo/Father;");
- //获取Java中对象数组Father属性objArrays的对象
- jobjectArray jobj_arr = (jobjectArray)env->GetObjectField(obj,fid_obj_arrays);
- //从对象数组中获取索引值为1的对象Father
- jobject jobj = env->GetObjectArrayElement(jobj_arr,1);
- //获取Father对象的class对象
- jclass clazz_father = env->GetObjectClass(jobj);
- //获取Father对象中的function方法的id
- jmethodID id_father_function = env->GetMethodID(clazz_father,"function","()V");
- //调用Father对象中的function方法
- env->CallVoidMethod(jobj,id_father_function);
- //在本地创建一个大小为10的对象数组,对象的初始化都是jobj,也就是方法的第三个参数
- jobjectArray jobj_arr_temp = env->NewObjectArray(10,env->GetObjectClass(jobj),jobj);
- //获取本地对象数组中第4个对象
- jobject jobj_temp = env->GetObjectArrayElement(jobj_arr_temp,3);
- //调用Father对象中的function方法
- env->CallVoidMethod(jobj_temp,id_father_function);
- }
- </span>
在Eclipse编译运行结果如下:
不要以为这就结束了,后面还有很多内容呀!
Java中JNI的使用详解第五篇:C/C++中操作Java中的数组的更多相关文章
- Java中JNI的使用详解第三篇:JNIEnv类型中方法的使用
转自: http://blog.csdn.net/jiangwei0910410003/article/details/17466369 上一篇说道JNIEnv中的方法的用法,这一篇我们就来通过例子来 ...
- Java中JNI的使用详解第四篇:C/C++中创建Java对象和String字符串对象及对字符串的操作方法
首先来看一下C/C++中怎么创建Java对象:在JNIEnv中有两种方法是用来创建Java对象的: 第一种方法: jobject NewObject(jclass clazz , jmethodI ...
- Java中JNI的使用详解第六篇:C/C++中的引用类型和Id的缓存
首先来看一下C/C++中的引用 从Java虚拟机创建的对象传到本地C/C++代码时会产生引用,根据Java的垃圾回收机制,只要有引用存在就不会触发该引用指向的Java对象的垃圾回收 第一.局部引用: ...
- Java中JNI的使用详解第二篇:JNIEnv类型和jobject类型的解释
上一篇说的是一个简单的应用,说明JNI是怎么工作的,这一篇主要来说一下,那个本地方法sayHello的参数的说明,以及其中方法的使用 首先来看一下C++中的sayHello方法的实现: JNIEXPO ...
- Java中JNI的使用详解第一篇:HelloWorld
转自: http://blog.csdn.net/jiangwei0910410003/article/details/17465085 今天开始研究JNI技术,首先还是老套路,输出一个HelloWo ...
- Orchard详解--第五篇 CacheManager
上一篇文章介绍了Orchard中的缓存,本篇主要针对CacheManager进行分析,CacheManager在Orchard中用于存储应用程序的配置信息以及框架内部的一些功能支持,包括整个拓展及拓展 ...
- JScript中的条件注释详解(转载自网络)
JScript中的条件注释详解-转载 这篇文章主要介绍了JScript中的条件注释详解,本文讲解了@cc_on.@if.@set.@_win32.@_win16.@_mac等条件注释语句及可用于条件编 ...
- Objective-C中 Self和 Super详解
Objective-C中 Self和 Super详解 Objective-C 中Self 和 Super 详解本文要介绍的内容,在 Objective-C 中的类实现中经常看到这两个关键字 self ...
- java中的io系统详解 - ilibaba的专栏 - 博客频道 - CSDN.NET
java中的io系统详解 - ilibaba的专栏 - 博客频道 - CSDN.NET 亲,“社区之星”已经一周岁了! 社区福利快来领取免费参加MDCC大会机会哦 Tag功能介绍—我们 ...
随机推荐
- 安装配置oh-my-zsh
1.下载安装iterm:https://www.iterm2.com/ 2.安装zsh 查看系统当前使用的shell $ echo $SHELL /bin/bash 查看系统是否安装了zsh $ ca ...
- Delphi如何获取一个字符串再另一个字符串中最后一次出现的位置
uses StrUtils; function ReversePos(SubStr, S: String): Integer; var i : Integer; begin i := Po ...
- json转换为map
// json转换为map public static Map parserToMap(String s) { Map map = new HashMap(); JSONObject json = J ...
- leetcood学习笔记-13
错误记录 class Solution: def romanToInt(self, s: str) -> int: d = {'I':1,'V':5,'X':10,'L':50,'C':100, ...
- 设置ll命令
ll 是 ls -l的别名,之所以 ll出现错误是因为没有定义别名. 如果要实现ll 命令,可以做如下操作: 1.编辑 ~./bashrc 添加 ls -l 的别名为 ll即可. vi /root/. ...
- IntelliJ IDEA2016.1 + maven 创建java web 项目[转]
最近开始使用idea 来写java项目了,这个很流行,相比Eclipse方便了很多.功能多了,相对应的使用的复杂度也较高了,因为网上很多的使用和创建项目的简单教程,都是基于老版本的,每个新版本都有不一 ...
- 搭建基于Linux6.3+Nginx1.2+PHP5+MySQL5.5的Web服务器全过程
http://blog.chinaunix.net/uid-20639775-id-154497.html
- xml配置离线约束的目的和ecplipse离线约束配置
正常情况下如果电脑已经联网的情况下,Spring的核心配置文件编写内容的时候是可以自动提示的,假设电脑如果离线情况下想要自动提示的话,就得配置离线约束文件. https://blog.csdn.net ...
- 2、Locust压力测试 实战
创建测试脚本 创建Test()类继承TaskSet类 创建beigong() 方法表示一个行为,访问北弓官网首页.用@task() 装饰该方法为一个任务.1表示一个Locust实例被挑选执行的权重,数 ...
- 7_springboot2.x开发热部署
概述:在开发中我们修改一个Java文件后想看到效果不得不重启应用,这导致大量时间花费,我们希望不重启应用的情况下,程序可以自动部署(热部署).有以下四种情况,如何能实现热部署. 1.模板引擎 在Spr ...