如何使用JNA将2D数组从Java传递到C.

时间:2014-01-20 18:22:32

标签: java c++ jna

我正在制作一个涉及使用JNA将2D数组从Java传递到C ++的项目。但是我在传递2D数组时遇到了问题。我能够毫无问题地传递一维数据。

Java代码 -

import com.sun.jna.Library;
import com.sun.jna.Native;

public class HelloWorld {
    public static void main(String argv[]) {
        CTest lib = CTest.ctest;
        int a[][] = {{2, 3}, {2, 4}};
        System.out.println(lib.helloFromC(a));
    }
    public interface CTest extends Library {
        CTest ctest = (CTest) Native.loadLibrary("cpptest", CTest.class);
        public int helloFromC(int x[][]);
    }

}

C ++代码

#include <stdio.h>
extern "C" __declspec(dllexport) 
int helloFromC(int a[][2]) {

    int i, sum = 0, j;
    for(i=0;i<2;i++) {
    for(j=0;j<2;j++) {
    printf("%d\n",a[i][j]);
        sum += a[i][j];}
    }
    printf("%d\n",sum);
    return sum;
}

当我编译这两个文件时,我没有错误,但是当我运行java文件时,我得到了

Exception in thread "main" java.lang.IllegalArgumentException: Unsupported array
 argument type: class [I
        at com.sun.jna.Function.convertArgument(Function.java:609)
        at com.sun.jna.Function.invoke(Function.java:297)
        at com.sun.jna.Library$Handler.invoke(Library.java:212)
        at $Proxy0.helloFromC(Unknown Source)
        at HelloWorld.main(HelloWorld.java:8)

1 个答案:

答案 0 :(得分:4)

在JNA中,当你在Java和C / C ++代码中都有原始的多维数组时,你需要在Java中使用1-D数组来映射到本机代码中的N-D数组。这是因为Java多维数组与本机C / C ++ N-D数组不同,因为每行的Java N-D数组的内存分配 NOT 是连续的。所以当你在Java中使用它时

int a[][] = {{2, 3}, {2, 4}};

您等效地使用两个new来创建二维数组:

int[] a = new int[2];
for(int i=0; i<2; i++) {
    a[i] = new int[2];
}

因此,为了解决Java和本机代码问题之间的这种不匹配,您必须使用Java中的1-D数组来模拟2-D数组,例如:

public static final int NUM_ROW = 2;
public static final int NUM_COL = 2;
int[] a = new int[NUM_ROW*NUM_COL];

对于1-D数组(i,j)th中的a元素,请使用

a[i*NUM_COL+j]

访问它。并且您可以将此1-D数组传递给JNA代码

System.out.println(lib.helloFromC(a));

JNA API overview中给出了有关多维数组如何在JNA代码中工作的详细说明。它说明了映射机制在JNA中的工作原理:

  

要映射本机多维数组,请使用单维Java   具有与完整本机数组相当的多个元素的数组