1、相关代码(有坑)
1.1、C相关代码如下:
cpp
结构体定义:
typedef struct {
char name[64]; /* 连接名称 */
unsigned char level; /* 安全性等级 */
} STRUCT_A;
调用方法定义:
FC_API void Initialize(
/* in */ STRUCT_A sttpConfigTable[],
/* out */ RETURN_CODE_TYPE *return_code
);
1.2、JNA相关代码如下:
java
JAVA结构体定义如下:
@Structure.FieldOrder({"name", "level"})
public class STRUCT_A extends Structure {
/* 连接名称 */
public byte[] name = new byte[64];
/* 安全性等级*/
public byte level;
public static class ByReferency extends STRUCT_A implements Structure.ByReference{}
public static class ByValue extends STRUCT_A implements Structure.ByValue{}
}
JNA调用方法声明:
void Initialize(STRUCT_A[] sttpConfigTable, IntByReference return_code);
1.3、业务代码
java
Java业务代码JNA调用dll如下:
IntByReference return_code = new IntByReference();
STRUCT_A[] sttpConfigTable = (STRUCT_A[]) new STRUCT_A().toArray(2);
//给sttpConfigTable中2个元素各属性赋值
byte[] nameBytes1 = "名称1".toBytes();
System.arraycopy(nameBytes1, 0, sttpConfigTable[0].name, 0, nameBytes1.length);
sttpConfigTable[0].level = 1;
byte[] nameBytes2 = "名称2".toBytes();
System.arraycopy(nameBytes2, 0, sttpConfigTable[1].name, 0, nameBytes2.length);
sttpConfigTable[0].level = 1;
//JNA调用
FcDll.Initialize(sttpConfigTable, return_code);
2、出现问题
结构体数组STRUCT_A[] sttpConfigTable从Java传递到C后,第一个元素取值正常,第二个元素取值异常,例如在C中取sttpConfigTable[1]的name属性乱码,不是"名称2"。
3、解决办法
在Java中查看结构体数组STRUCT_A[] sttpConfigTable的大小为136(每个元素68字节),但是在C中每个元素是65字节。因为Java中结构体的内存对齐方式默认是按照4字节字节对齐,也就是4字节的整数倍,但是C中默认是1字节对齐,因此需要将Java结构体STRUCT_A的内存方式设置成和C相同:1字节对齐。代码修改如下(在结构体STRUCT_A的构造行数中调用"super(ALIGN_NONE)"即可):
java
@Structure.FieldOrder({"name", "level"})
public class STRUCT_A extends Structure {
/* 连接名称 */
public byte[] name = new byte[64];
/* 安全性等级*/
public byte level;
public STRUCT_A() {
super(ALIGN_NONE);
}
public static class ByReferency extends STRUCT_A implements Structure.ByReference{}
public static class ByValue extends STRUCT_A implements Structure.ByValue{}
}