JNI中基本类型数组的传递方法(无需拷贝数据!!!)
编程技术  /  houtizong 发布于 3年前   264
GetIntArrayElements();//貌似得到的是副本,要拷贝数据ReleaseIntArrayElements();//对应上面的函数的释放资源的函数env->GetPrimitiveArrayCritical();//貌似得到的是指向原数据的指针env->ReleasePrimitiveArrayCritical();////对应上面的函数的释放资源的函数
package tests;import java.util.Arrays;public class TestJNIArray {static{System.loadLibrary("TestJNIArray");}public static native void addOne(int[] ints);//数组元素+1public static native void addOne(double[] ints);//数组元素+1,为了测试,C++中循环了5次public static native int[] getNewArray(int size,int initValue);//生成初始值为initValue的数组,数组长度为sizepublic static void main(String[] args) throws InterruptedException {int n=20;final int[] a=new int[n];for (int i = 0; i < a.length; i++) {a[i]=i;}if(n<50)System.out.println(Arrays.toString(a));addOne(a);if(n<50)System.out.println(Arrays.toString(a));final double d[]=new double[n];for (int i = 0; i < d.length; i++) {d[i]=i;}//addOne(d);if(n<50)System.out.println(Arrays.toString(d));new Thread(new Runnable() {@Overridepublic void run() {addOne(d);}}).start();for (int i = 0; i < 200; i++) {Thread.sleep(20);System.out.println(Arrays.toString(d));if(d[d.length-1]-d[0]!=n-1)System.out.println("检测到C++端更新数据中");//看看能否找到在C++更新数组时Java端又读取数据的情况}int[] b=getNewArray(2, 9);System.out.println(Arrays.toString(b));int[] c=getNewArray(0, 9);System.out.println(Arrays.toString(c));}}
#include "tests_TestJNIArray.h"#include <windows.h>#include <time.h>#include <iostream>#include <string>class Timer{private:clock_t time;public:Timer(){time=clock();}clock_t getElapsedTime(){return clock()-time;}clock_t getElapsedTimeAndRestart(){clock_t tmp=time;time=clock();return time-tmp;}void restart(){time=clock();}int getCLOCKS_PER_SEC(){return CLOCKS_PER_SEC;}};JNIEXPORT void JNICALL Java_tests_TestJNIArray_addOne___3I( JNIEnv * env, jclass, jintArray intArray) {jboolean b;Timer timer; jint* pint=env->GetIntArrayElements(intArray,&b);//获取指针!!第二个参数可为NULLstd::string str=b?"true":"false";long t=timer.getElapsedTimeAndRestart();std::cout<<"GetIntArrayElements()耗时: "<<t<<"\t为副本?"<<str<<std::endl;//很悲剧,自己得到的是副本(copy了一份,速度慢啊) jsize size=env->GetArrayLength(intArray);for (int i=0; i<size; i++)pint[i]+=1; env->ReleaseIntArrayElements(intArray,pint,0);//释放~~~//对于最后一个参数(如果指针指向的数组为副本时,否则该参数不起作用)// 0copy back the content and free the elems buffer// JNI_COMMITcopy back the content but do not free the elems buffer// JNI_ABORTfree the buffer without copying back the possible changesstd::cout<<"从c++程序返回~"<<std::endl;}JNIEXPORT void JNICALL Java_tests_TestJNIArray_addOne___3D( JNIEnv * env, jclass, jdoubleArray intArray){jboolean b;Timer timer;double* pd=(double*)env->GetPrimitiveArrayCritical(intArray,&b);//if(pd==NULL)return;理论上应该检查!!!!long t=timer.getElapsedTimeAndRestart();jsize size=env->GetArrayLength(intArray);std::string str=b?"true":"false";std::cout<<"GetPrimitiveArrayCritical()耗时: "<<t<<"\t为副本?"<<str<<std::endl;//这次是原始数据了,happy啊!!!for(int j=0;j<5;j++){ //验证一下,Java中的数据也在更新!!!Sleep(1000);for (int i=0; i<size; i++){pd[i]+=1;Sleep(10); }}env->ReleasePrimitiveArrayCritical(intArray,pd,0);//别忘了释放~~~虽然不知道不释放有什么问题。。。std::cout<<"从c++程序返回~"<<std::endl;}JNIEXPORT jintArray JNICALL Java_tests_TestJNIArray_getNewArray( JNIEnv * env, jclass, jint size, jint initValue ) { jintArray intArray=env->NewIntArray(size);jboolean* pb; //jint* pint=env->GetIntArrayElements(intArray,pb);jint* pint=new jint[size];std::string str=*pb?"true":"false";std::cout<<"自己生成的Java数组,取出指针,为副本?"<<str<<std::endl; for (int i=0; i<size; i++) { pint[i]=initValue; } env->ReleaseIntArrayElements(intArray,pint,JNI_COMMIT);delete []pint; //如果使用GetIntArrayElements()得到该指针则不可以!!!否则Java直接崩溃了 return intArray;}
请勿发布不友善或者负能量的内容。与人为善,比聪明更重要!
技术博客集 - 网站简介:
前后端技术:
后端基于Hyperf2.1框架开发,前端使用Bootstrap可视化布局系统生成
网站主要作用:
1.编程技术分享及讨论交流,内置聊天系统;
2.测试交流框架问题,比如:Hyperf、Laravel、TP、beego;
3.本站数据是基于大数据采集等爬虫技术为基础助力分享知识,如有侵权请发邮件到站长邮箱,站长会尽快处理;
4.站长邮箱:[email protected];
文章归档
文章标签
友情链接