在C中使用大型数组时的EXCEPTION_ACCESS_VIOLATION

时间:2014-01-13 18:37:13

标签: c arrays matlab java-native-interface mat-file

这是我坚持的另一个问题。首先,我正在尝试读取包含从PicoScope 6导出的信息的4级matlab文件,它从文件A,Tstart,Tinterval和Length读取四个数组。第一个数组是迄今为止最大的数组,它包含1000004个值,但其他三个只包含一个值。当我执行下面的代码时,它成功读取文件,将其存储到多维数组中,但当我尝试使用数组时失败

#include <jni.h>
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <matio.h>
#include "ReadMatFile.h"

//macros
#define getLength(x)  (sizeof(x) / sizeof(x[0]))

//variables
double *dataMatrix[4];
int innerSize[getLength(dataMatrix)];

//functions
jobjectArray convertToArray(JNIEnv *env, double **data, int length1D,
        int *length2D);

JNIEXPORT jdoubleArray JNICALL Java_ReadMatFile_readMatFile(JNIEnv *env,
        jobject object, jstring str) {
    const char *fileName = (*env)->GetStringUTFChars(env, str, 0);
    mat_t *matfp;
    matvar_t *matvar;
    matfp = Mat_Open(fileName, MAT_ACC_RDONLY | MAT_FT_MAT4);
    if ( NULL == matfp) {
        fprintf(stderr, "Error opening MAT file %s\n", fileName);
        return NULL;
    }
    int i = 0;
    while ( NULL != (matvar = Mat_VarReadNext(matfp))) {
        double *data = (double*) (matvar->data);
        dataMatrix[i] = data;
        innerSize[i] = (int) matvar->nbytes / matvar->data_size;
        Mat_VarFree(matvar);
        matvar = NULL;
        i++;
    }
    Mat_Close(matfp);
    (*env)->ReleaseStringUTFChars(env, str, fileName);
    int s;
    for(s = 0; s < innerSize[0]; s++)
        printf("A[%d] = %e\n", s, dataMatrix[0][s]); /* Fails here */
    return NULL;
    //return convertToArray(env, dataMatrix, getLength(dataMatrix) ,innerSize);
}

jobjectArray convertToArray(JNIEnv *env, double **data, int length1D,
        int *length2D) {
    jsize outerSize = (jsize) length1D;
    jclass class = (*env)->FindClass(env, "[D");
    jobjectArray outer = (*env)->NewObjectArray(env, outerSize, class, 0);
    jsize i;
    for (i = 0; i < outerSize; i++) {
        jsize innerSize = (jsize) length2D[i];
        jdoubleArray inner = (*env)->NewDoubleArray(env, innerSize);
        (*env)->SetDoubleArrayRegion(env, inner, 0, innerSize, data[i]);
        (*env)->SetObjectArrayElement(env, outer, i, inner);
        (*env)->DeleteLocalRef(env, inner);
    }
    return outer;
}

这是什么原因?它在我运行此应用程序时创建minidump。阵列太大了吗?

修复以及错误的解释将非常感谢!

先谢谢你们。

1 个答案:

答案 0 :(得分:2)

我怀疑您的问题出现在以下代码中:

double *data = (double*) (matvar->data);
dataMatrix[i] = data;
innerSize[i] = (int) matvar->nbytes / matvar->data_size;
Mat_VarFree(matvar); // whoopsie

如果Mat_VarFree执行我认为的操作,matvar->data不再是有效指针,意味着dataMatrix[i]不再是有效指针,因此崩溃。

我认为你打算做的事情更像是

innerSize[i] = matvar->nbytes / matvar->data_size;
dataMatrix[i] = malloc( sizeof *dataMatrix[i] * innerSize[i] );
if ( dataMatrix[i] )
  memcpy( dataMatrix[i], matvar->data, matvar->nbytes );
Mat_VarFree( matvar );

即,在matvar->data中创建数据的本地副本,并将其保存到dataMatrix。在原始代码中,您复制的只是一个指针值;您从未创建过单独的数据副本。