JNA结构体数组

本文主要讲述使用JNA模拟结构体并将结构体数组作为参数传递给对应的方法。

C语言结构体定义如下:

typedef struct Rect
{
	int top;
	int bottom;
	int left;
	int right;
} RECT;

JNA模拟该结构体:

需要引入:

import com.sun.jna.*;
import com.sun.jna.ptr.*;

    //Rect结构体
    public static class Rect extends Structure {
        //Structure子类中的公共字段的顺序,必须与C语言中的结构的顺序一致,否则会报错!
        public int top;
        public int bottom;
        public int left;
        public int right;

        public static class ByReference extends Rect implements Structure.ByReference {}
        public static class ByValue extends Rect implements Structure.ByValue {}

        @Override
        protected List getFieldOrder() {
            return Arrays.asList(new String[]{"top", "bottom", "left", "right"});
        }
    }
现在要给方法传递一个结构体数组对象,如何做呢?

C语言函数:

//rects:结构体数组,len:数组长度
void function(RECT* rects, int len);

JNA模拟如下:

void function(Rect[] rects,int len);

创建数组及调用方法如下:

int len = 5;
//定义数组
Rect[] array = (Rect[])new Rect().toArray(len);
function(array, len);

注意,这里创建数组使用了JNA的toArray()方法,而不是Java常规创建数组的方法,是因为内存空间在java中是不连续的,jna定义数组是需要使用toArray方法,这样实例化出来的数组内存空间是连续的。

其实这里主要是创建结构体数组的地方要注意,如果使用下面的方式创建,那就会产生一个空的数组,申请不到空间:

int len = 5;
Rect[] array = new Rect[len];
这个代码执行后,array=null,也就是创建数组失败了!

原因:内存空间在java中是不连续的,jna定义数组是需要使用toArray方法,这样实例化出来的数组内存空间是连续的,否则数据第一条是正确的,其他是错位的。

详见另一篇文章:http://tcspecial.iteye.com/blog/1665583

感谢一楼网友解答!




  • 4
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
在使用JNA调用SO库时,需要注意传递参数的数据类型和内存布局,以确保传递的参数正确无误。 对于一个结构体数组,我们可以使用JNA的`Structure`类来表示,同时在定义结构体时,需要使用`@Field`注解来指定字段在内存中的偏移量和数据类型。 对于整型二维数组中的负数问题,可以在定义结构体时,使用无符号整型`int`来表示,这样可以确保数据不会出现负数。 具体的代码实现可以参考以下示例: ```java import com.sun.jna.Structure; import com.sun.jna.ptr.PointerByReference; import java.util.Arrays; import java.util.List; public class MyLibrary { public static class MyStruct extends Structure { public int[][] intArray; public MyStruct(int[][] intArray) { this.intArray = intArray; } public static class ByReference extends MyStruct implements Structure.ByReference { public ByReference(int[][] intArray) { super(intArray); } } @Override protected List getFieldOrder() { return Arrays.asList("intArray"); } @Override protected java.util.Map<String, Object> getFieldOrderMap() { java.util.Map<String, Object> map = super.getFieldOrderMap(); map.put("intArray", int[][].class); return map; } public static MyStruct[] toArray(PointerByReference pbr, int length) { return (MyStruct[])Structure.toArray(pbr, length); } } public static native void myFunction(MyStruct[] structArray, int length); } ``` 在上述代码中,`MyStruct`表示一个结构体,其中包含一个整型二维数组`intArray`。`MyStruct.ByReference`表示一个指向结构体的指针,用于传递给SO库中的函数。 在`MyStruct`中,我们重写了`getFieldOrder`和`getFieldOrderMap`方法,用于指定结构体中的字段顺序和数据类型。 在`MyStruct.toArray`方法中,我们使用`Structure.toArray`将指向结构体数组的指针转换为Java中的结构体数组。 在使用JNA调用SO库的过程中,需要注意使用相应的数据类型和内存布局,以确保传递的参数正确无误。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值