如何传递结构数组以便在JNA中获得结果

时间:2014-06-10 11:46:07

标签: java c jna

我在C中有以下功能:

EXPORT int GetInfo(MyObject* &myObjects);

typedef struct MyObject
{
    char info1[1025];
    unsigned long sizeF;
    char info2[20];
};

然后我调用:

MyObject* list1;
int count = GetInfo(list1);

并迭代list1以获取每个MyObject对象的信息(count - > list1中的元素数)。

现在,我正在尝试在JNA中制作对应物。因此,我有:

int GetInfo(PointerByReference myObjets);

public class MyObject extends Structure {

   public static class ByReference extends MyObject implements Structure.ByReference {
   }
   public String info1;
   public NativeLong sizeF;
   public String info2;

   public MyObject() {
   }

   public MyObject(Pointer pointer) {
      super(pointer);
   }

   @Override
   protected List getFieldOrder() {
      return Arrays.asList(new String[]{"info1", "sizeF", "info2"});
   }
}

然后:

PointerByReference ptrRef = new PointerByReference();
int count = myLib.GetInfo(ptrRef);
if (count > 0) {
   Pointer pointer = ptrRef.getValue();
   MyObject myObject = new MyObject(pointer);
   MyObject[] myObjects = (MyObject[]) myObject.toArray(count);
}

不幸的是,myObjects中的所有字段都有默认值(null / 0)。 我也尝试过:

int GetInfo(MyObject.ByReference myObjets);
MyObject.ByReference byRef = new PointerByReference();
int count = myLib.GetInfo(byRef);
if (count > 0) {
   MyObject[] myObjects = (MyObject[]) byRef.toArray(count);
}

在这种情况下,只填充myObjects数组的第一个元素中的第一个字段。其余的都有默认值。

我应该怎么做才能获得一个包含所有字段的MyObjects数组。

1 个答案:

答案 0 :(得分:1)

前段时间我找到了解决方案。我不知道这是否明智,但它对我有用。所以,如果有人会有类似的问题,那么你就去吧:

在MyObject类中,我添加了两个方法:

static MyObject[] fromArrayPointer(Pointer pointer, int numberResults) {
    MyObject[] arr = new MyObject[numberResults];
    int offset = 0;
    for (int i = 0; i < numberResults; i++) {
        arr[i] = fromPointer(pointer, offset);
        offset += <size of structure>;
    }
    return arr;
}

static MyObject fromPointer(Pointer pointer, int offset) {  
    MyObject inst = new MyObject();
    inst.info1= pointer.getString(offset);
    offset += 1025;
    inst.sizeF = pointer.getNativeLong(offset);
    offset += 4; // long but 4 bytes because of machine
    inst.info2 = pointer.getString(offset);
    offset += 20;
    return inst;
}

老实说,你必须试验那些数字和大小的结构。请记住这里的数据对齐问题。

然后,你有:

if (count > 0) {
   MyObject[] myObjects = MyObject.fromArrayPointer(ptrRef.getValue(), count);
}