如何 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 语法)