JNA内存对齐导致的结构体数组传输错误

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{}
}
相关推荐
地平线开发者1 天前
SparseDrive 模型导出与性能优化实战
算法·自动驾驶
董董灿是个攻城狮1 天前
大模型连载2:初步认识 tokenizer 的过程
算法
地平线开发者1 天前
地平线 VP 接口工程实践(一):hbVPRoiResize 接口功能、使用约束与典型问题总结
算法·自动驾驶
罗西的思考1 天前
AI Agent框架探秘:拆解 OpenHands(10)--- Runtime
人工智能·算法·机器学习
HXhlx1 天前
CART决策树基本原理
算法·机器学习
Wect1 天前
LeetCode 210. 课程表 II 题解:Kahn算法+DFS 双解法精讲
前端·算法·typescript
颜酱1 天前
单调队列:滑动窗口极值问题的最优解(通用模板版)
javascript·后端·算法
Gorway1 天前
解析残差网络 (ResNet)
算法
拖拉斯旋风1 天前
LeetCode 经典算法题解析:优先队列与广度优先搜索的巧妙应用
算法
Wect1 天前
LeetCode 207. 课程表:两种解法(BFS+DFS)详细解析
前端·算法·typescript