如何 return 一个从 C 到 Java 的 3D 数组?
How to return a 3D array from C to Java?
让我介绍一下我的情况:
问题:在 c 中创建一个 3D 数组并通过 JNI 将其传递给 java。
进展:到目前为止我已经尝试了很多选择,这就是我到现在为止的结果:
#include <jni.h>
#include <stdio.h>
#include<stddef.h>
#include<com_example_c2java_MainActivity.h>
JNIEXPORT jobjectArray JNICALL Java_com_example_c2java_MainActivity_getArray
(JNIEnv *env, jobject obj){
double a[2][2][2];
int i,j,k;
double x = 0;
jclass doubleClass = (*env)->FindClass(env,"Java/lang/Double");//
jobjectArray ret = (*env)->NewObjectArray(env, 2, doubleClass, NULL);
for( i = 0; i<2; i++){
for( j = 0; j<2; j++){
for( k = 0; k<2; k++){
x+=1;
a[i][j][k] = x;
}
}
}
for( i = 0; i<2; i++){
jdoubleArray tmp1 = (*env)->NewDoubleArray(env,2);;
jdoubleArray dim2 = (*env)->NewDoubleArray(env,2);
for( j = 0; j<2; j++){
jdouble tmp2[256];
jdoubleArray dim3 = (*env)->NewDoubleArray(env,2);
for( k = 0; k<2; k++){
tmp2[k] = a[i][j][k];
}
(*env)->SetDoubleArrayRegion(env,tmp1 , j, 2, tmp2);
}
(*env)->SetDoubleArrayRegion(env,dim2, 0, 2, tmp1);
(*env)->SetObjectArrayElement(env,ret, i, dim2);
(*env)->DeleteLocalRef(env, dim2);
}
return ret;
}
java 边:
package com.example.c2java;
import android.support.v7.app.ActionBarActivity;
import android.annotation.TargetApi;
import android.os.Build;
import android.os.Bundle;
import android.util.Log;
import android.view.Menu;
import android.view.MenuItem;
@TargetApi(Build.VERSION_CODES.HONEYCOMB)
public class MainActivity extends ActionBarActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
double[][][] arr = getArray();
Log.i("ffff","-----------------------------"+arr[0][0][0]);
}
static{
System.loadLibrary("C2Java");
}
public native double[][][] getArray();
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.main, menu);
return true;
}
}
错误:启动应用程序会出现以下 logcat 错误:
02-19 16:20:29.084: E/dalvikvm(20707): VM aborting
02-19 16:20:29.084: A/libc(20707): Fatal signal 11 (SIGSEGV) at 0xdeadd00d (code=1), thread 20707 (.example.c2java)
解决方案:需要您的帮助才能找到它:)
你好像误解了涉及的类型是什么。
首先,Java/lang/Double
与原始类型double
不一样。其次,ret
不应该是double
的数组,应该是double
的数组的数组。
你想做的是这样的:
jclass doubleArrayArrayClass = env->FindClass("[[D");
jclass doubleArrayClass = env->FindClass("[D");
// Create an array of arrays of arrays of double
jobjectArray ret = env->NewObjectArray(2, doubleArrayArrayClass, NULL);
for( i = 0; i<2; i++){
for( j = 0; j<2; j++){
for( k = 0; k<2; k++){
x+=1;
a[i][j][k] = x;
}
}
}
for( i = 0; i<2; i++){
// ret[i] is an array of arrays of double
jobjectArray dim2 = env->NewObjectArray(2, doubleArrayClass, NULL);
for( j = 0; j<2; j++) {
// ret[i][j] is an array of double
jdoubleArray dim1 = env->NewDoubleArray(2);
jdouble tmp[256];
for( k = 0; k<2; k++){
tmp[k] = a[i][j][k];
}
env->SetDoubleArrayRegion(dim1 , 0, 2, tmp);
env->SetObjectArrayElement(dim2, j, dim1);
env->DeleteLocalRef(dim1);
}
env->SetObjectArrayElement(ret, i, dim2);
env->DeleteLocalRef(dim2);
}
(我对 JNI 调用使用 C++ 语法,因此如果必须使用 C 编译器,则必须将其改回 C 语法)
让我介绍一下我的情况:
问题:在 c 中创建一个 3D 数组并通过 JNI 将其传递给 java。
进展:到目前为止我已经尝试了很多选择,这就是我到现在为止的结果:
#include <jni.h>
#include <stdio.h>
#include<stddef.h>
#include<com_example_c2java_MainActivity.h>
JNIEXPORT jobjectArray JNICALL Java_com_example_c2java_MainActivity_getArray
(JNIEnv *env, jobject obj){
double a[2][2][2];
int i,j,k;
double x = 0;
jclass doubleClass = (*env)->FindClass(env,"Java/lang/Double");//
jobjectArray ret = (*env)->NewObjectArray(env, 2, doubleClass, NULL);
for( i = 0; i<2; i++){
for( j = 0; j<2; j++){
for( k = 0; k<2; k++){
x+=1;
a[i][j][k] = x;
}
}
}
for( i = 0; i<2; i++){
jdoubleArray tmp1 = (*env)->NewDoubleArray(env,2);;
jdoubleArray dim2 = (*env)->NewDoubleArray(env,2);
for( j = 0; j<2; j++){
jdouble tmp2[256];
jdoubleArray dim3 = (*env)->NewDoubleArray(env,2);
for( k = 0; k<2; k++){
tmp2[k] = a[i][j][k];
}
(*env)->SetDoubleArrayRegion(env,tmp1 , j, 2, tmp2);
}
(*env)->SetDoubleArrayRegion(env,dim2, 0, 2, tmp1);
(*env)->SetObjectArrayElement(env,ret, i, dim2);
(*env)->DeleteLocalRef(env, dim2);
}
return ret;
}
java 边:
package com.example.c2java;
import android.support.v7.app.ActionBarActivity;
import android.annotation.TargetApi;
import android.os.Build;
import android.os.Bundle;
import android.util.Log;
import android.view.Menu;
import android.view.MenuItem;
@TargetApi(Build.VERSION_CODES.HONEYCOMB)
public class MainActivity extends ActionBarActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
double[][][] arr = getArray();
Log.i("ffff","-----------------------------"+arr[0][0][0]);
}
static{
System.loadLibrary("C2Java");
}
public native double[][][] getArray();
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.main, menu);
return true;
}
}
错误:启动应用程序会出现以下 logcat 错误:
02-19 16:20:29.084: E/dalvikvm(20707): VM aborting
02-19 16:20:29.084: A/libc(20707): Fatal signal 11 (SIGSEGV) at 0xdeadd00d (code=1), thread 20707 (.example.c2java)
解决方案:需要您的帮助才能找到它:)
你好像误解了涉及的类型是什么。
首先,Java/lang/Double
与原始类型double
不一样。其次,ret
不应该是double
的数组,应该是double
的数组的数组。
你想做的是这样的:
jclass doubleArrayArrayClass = env->FindClass("[[D");
jclass doubleArrayClass = env->FindClass("[D");
// Create an array of arrays of arrays of double
jobjectArray ret = env->NewObjectArray(2, doubleArrayArrayClass, NULL);
for( i = 0; i<2; i++){
for( j = 0; j<2; j++){
for( k = 0; k<2; k++){
x+=1;
a[i][j][k] = x;
}
}
}
for( i = 0; i<2; i++){
// ret[i] is an array of arrays of double
jobjectArray dim2 = env->NewObjectArray(2, doubleArrayClass, NULL);
for( j = 0; j<2; j++) {
// ret[i][j] is an array of double
jdoubleArray dim1 = env->NewDoubleArray(2);
jdouble tmp[256];
for( k = 0; k<2; k++){
tmp[k] = a[i][j][k];
}
env->SetDoubleArrayRegion(dim1 , 0, 2, tmp);
env->SetObjectArrayElement(dim2, j, dim1);
env->DeleteLocalRef(dim1);
}
env->SetObjectArrayElement(ret, i, dim2);
env->DeleteLocalRef(dim2);
}
(我对 JNI 调用使用 C++ 语法,因此如果必须使用 C 编译器,则必须将其改回 C 语法)