我在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数组。
答案 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);
}