首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >问答首页 >JNA:如何在结构中指定可变长度(0+)数组?

JNA:如何在结构中指定可变长度(0+)数组?
EN

Stack Overflow用户
提问于 2020-12-13 23:23:08
回答 1查看 370关注 0票数 3

示例结构,其中count是数组中的字节数,可以是0。我希望在Java中分配新实例,并读取本机分配的实例。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
    public class VarArray extends Structure {
        public byte dummy0;
        public short dummy1;
        public int count;
        public byte[] array;
    }

array = new byte[0]中使用Structure是不允许的。

如果计数为0,则声明默认的array = new byte[1]将从未分配的地址读取。

删除array字段可以读取,因为我可以访问指针偏移量Structure.size()编辑:不正确的字节,取决于填充。但是,为了分配一个新实例,我需要手动确定字段大小和对齐填充,以便分配正确的内存大小。

我有一个使用两种类型的解决方案-一种是对本机分配的和0计数的array分配的实例没有array,另一种是带有array for Java分配的1+计数实例的子类型。这似乎相当臃肿,特别是与所需的锅炉板代码。

有更好的办法吗?

或者是一种简单的方法来计算字段大小和对齐方式,这样一种类型就足够了?

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
import java.util.Arrays;
import java.util.List;
import com.sun.jna.Pointer;
import com.sun.jna.Structure;

public class JnaStructTester {

    /**
     * For native-allocated, and 0-count JNA-allocated instances.
     */
    public static class VarArray extends Structure {
        public byte dummy0;
        public short dummy1;
        public int count;

        public VarArray() {}

        public VarArray(Pointer p) {
            super(p);
        }

        public byte[] getArray() {
            byte[] array = new byte[count];
            if (count > 0) {
                int offset = size();
                getPointer().read(offset, array, 0, count);
            }
            return array;
        }
        
        @Override
        protected List<String> getFieldOrder() {
            return List.of("dummy0", "dummy1", "count");
        }
    }
    
    /**
     * For 1+ count JNA-allocated instances.
     */
    public static class VarArrayX extends VarArray {
        public byte[] array;

        public VarArrayX() {}

        @Override
        public byte[] getArray() {
            return array;
        }
        
        @Override
        protected List<String> getFieldOrder() {
            return List.of("dummy0", "dummy1", "count", "array");
        }
    }
    
    public static void main(String[] args) {
        var va0 = new VarArrayX();
        va0.dummy0 = (byte) 0xef;
        va0.dummy1 = (short) 0xabcd;
        va0.count = 7;
        va0.array = new byte[] { 1, 2, 3, 4, 5, 6, 7 };
        va0.write();
        
        var va1 = new VarArray();
        va1.dummy0 = (byte) 0xab;
        va1.dummy1 = (short) 0xcdef;
        va1.write();
        
        print(new Pointer(Pointer.nativeValue(va0.getPointer())));
        print(new Pointer(Pointer.nativeValue(va1.getPointer())));
    }
    
    private static void print(Pointer p) {
        var va = new VarArray(p);
        va.read();
        System.out.println(va);
        System.out.println("byte[] array=" + Arrays.toString(va.getArray()));
        System.out.println();
    }
}

输出:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
JnaStructTester$VarArray(native@0x7fb6835524b0) (8 bytes) {
  byte dummy0@0=ffffffef
  short dummy1@2=ffffabcd
  int count@4=7
}
byte[] array=[1, 2, 3, 4, 5, 6, 7]

JnaStructTester$VarArray(native@0x7fb683551210) (8 bytes) {
  byte dummy0@0=ffffffab
  short dummy1@2=ffffcdef
  int count@4=0
}
byte[] array=[]

(我使用的是相当老的JNA版本4.2.2)

更新(2020-01-07)

感谢Daniel的建议,这里有一个很好的解决方案。

基于数组是否为空,动态修改字段列表是不可能的。当不包括变量数组字段时,布局将静态缓存,因此具有非空数组的未来实例将失败。相反:

  1. 该数组在ensureAllocated()中进行调整,以避免空数组错误。
  2. 仅当数组为非空时,writeField()才写入该字段。
  3. readField()从count中设置数组大小,它已经被读取,如果计数为0,则跳过读取数组。

通过在readField()中将数组设置为正确的大小,将自动填充完整的数组,无需手动创建。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
import java.util.Arrays;
import java.util.List;
import com.sun.jna.Pointer;
import com.sun.jna.Structure;

public class JnaStructTester {

    public static class VarArray extends Structure {
        public short dummy0;
        public int dummy1;
        public byte count;
        public byte[] array = new byte[0];

        public VarArray() {}

        public VarArray(byte[] array) {
            this.count = (byte) array.length;
            this.array = array;
        }

        public VarArray(Pointer p) {
            super(p);
        }

        @Override
        protected void ensureAllocated() {
            if (count == 0) array = new byte[1];
            super.ensureAllocated();
            if (count == 0) array = new byte[0];
        }

        @Override
        protected void writeField(StructField structField) {
            if (structField.name.equals("array") && count == 0) return;
            super.writeField(structField);
        }

        @Override
        protected Object readField(StructField structField) {
            if (structField.name.equals("array")) {
                array = new byte[count];
                if (count == 0) return null;
            }
            return super.readField(structField);
        }

        @Override
        protected List<String> getFieldOrder() {
            return List.of("dummy0", "dummy1", "count", "array");
        }
    }

    public static void main(String[] args) {
        var va0 = new VarArray(new byte[] { 1, 2, 3, 4, 5, 6, 7 });
        va0.dummy0 = 0x4321;
        va0.dummy1 = 0xabcdef;
        va0.write();

        var va1 = new VarArray();
        va1.dummy0 = 0x4321;
        va1.dummy1 = 0xabcdef;
        va1.write();
        
        print(new Pointer(Pointer.nativeValue(va0.getPointer())));
        print(new Pointer(Pointer.nativeValue(va1.getPointer())));
    }

    private static void print(Pointer p) {
        var va = new VarArray(p);
        va.read();
        System.out.println(va);
        System.out.println("byte[] array=" + Arrays.toString(va.array));
        System.out.println();
    }
}

输出:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
JnaStructTester$VarArray(native@0x7fd85cf1ffb0) (12 bytes) {
  short dummy0@0=4321
  int dummy1@4=abcdef
  byte count@8=7
  byte array[7]@9=[B@4f2410ac
}
byte[] array=[1, 2, 3, 4, 5, 6, 7]

JnaStructTester$VarArray(native@0x7fd85cf20690) (12 bytes) {
  short dummy0@0=4321
  int dummy1@4=abcdef
  byte count@8=0
  byte array[0]@9=[B@722c41f4
}
byte[] array=[]

我只在我的狭隘用例中测试过这一点,如果在Structure上调用其他方法,它可能无法工作。

EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2020-12-14 08:59:35

我确实认为有两个结构的解决方案是一个合理的解决方案,数组版本扩展了另一个结构,但简单地添加新字段。您对“样板膨胀”的关注大大减少了JNA5.X,它有一个@FieldOrder注释,这大大减少了样板。您的结构将相当简单:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
@FieldOrder({"dummy0", "dummy1", "count"})
public class VarArray extends Structure {
    public byte dummy0;
    public short dummy1;
    public int count;
}

@FieldOrder({"dummy0", "dummy1", "count", "array"})
public class VarArrayX extends VarArray  {
    public byte[] array = new byte[1];

    public VarArrayX(int arraySize) {
        array = new byte[arraySize];
        super.count = arraySize;
        allocateMemory();
    }
}

除了添加构造函数(如果要使用指针进行初始化)之外,这应该足够了。

但是,您可以通过对FieldOrder进行相同的修改来执行单一结构版本。JNA的内存分配取决于使用getFieldList()getFieldOrder()方法定义的字段顺序(新注释删除了模板要求)。

可以将数组分配保留在结构中,但更改getFieldOrder()覆盖以跳过定义数组的字段,在数组为零的情况下,对getFieldList()也是如此。在JNA的Linux映射中编写Sysinfo结构时,我必须处理这种情况。主要结构包括这一领域:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public byte[] _f = new byte[PADDING_SIZE];

但是,getFieldList()覆盖包括:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
if (PADDING_SIZE == 0) {
    Iterator<Field> fieldIterator = fields.iterator();
    while (fieldIterator.hasNext()) {
        Field field = fieldIterator.next();
        if ("_f".equals(field.getName())) {
            fieldIterator.remove();
        }
    }
}

getFieldOrder()包括:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
if (PADDING_SIZE == 0) {
    fieldOrder.remove("_f");
}

类似的条件将删除同一文件中的_f_unused结构中的Statvfs字段。

在您的结构中,只要您在实例化结构之前知道count,这样的东西就可以工作(未经测试):

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public static class VarArray extends Structure {
    public byte dummy0;
    public short dummy1;
    public int count;
    public byte[] array = new byte[1];

    public VarArray(int arraySize) {
        array = new byte[arraySize];
        count = arraySize;
        allocateMemory();
    }

    @Override
    protected List<String> getFieldOrder() {
        if (count == 0) {
            return List.of("dummy0", "dummy1", "count");
        } else {
            return List.of("dummy0", "dummy1", "count", "array");
        }
    }

    // do the same for getFieldList()
}
票数 3
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/65285060

复制
相关文章
将oracle端口1521共享_oracle如何开放1521端口
在注册表:HKEY_LOCAL_MACHINESOFTWAREORACLEHOME0上新建一个字符串值:USE_SHARED_SOCKET=true.如果安装了多个目录,则每个类似的目录都要设置:HKEY_LOCAL_MACHINESOFTWAREORACLEHOMEx (x目录编号)
全栈程序员站长
2022/06/28
1.8K0
将hello程序作为驱动程序编译进系统内核
恩,可能是我比较愚钝,一个内核编译搞了一天,各种问题,各种bug,几度无奈,也是因为我突发奇想,并没有按照原来的那种操作,我直接把helloworld程序放到内核模块中编译成了一个驱动程序,虽然其中遇到了不知道多少的问题,不过最终是个完美的结局,给自己点个赞! 好了,废话不多说,直接开始还原我的helloworld驱动内核程序编译流程。
roobtyan
2019/02/21
1.5K0
将hello程序作为驱动程序编译进系统内核
WordPress 建立数据库连接时出错
WordPress 网站,需要在一个运行PHP 7.4或更高版本;数据库软件可采用MySQL 5.6或更高版本的服务器中才能运行的。
.T.
2022/02/19
5.3K0
WordPress 建立数据库连接时出错
wordpress建立数据库连接时出错
1、表单信息正确无误 2、安全组配置已更改 3、数据库远程连接已开启 4、数据库在3306端口正常运行并可连接 5、本地无法远程连接数据库
李玺
2022/08/03
3.2K0
wordpress建立数据库连接时出错
将 windows 磁盘作为共享盘挂载到 Linux 上
在 Linux 处理组学数据,硬盘真是经不起消耗的东西。而本地办公的电脑主要以开浏览器、看文献、交流、做PPT等为主,除了C盘小的时候容易炸,其他磁盘使用量很少。所以我最近遇到了工作站没法装大点的测序数据,但 PC 却有好几个盘上 T 的空闲。这篇文章就简单做个笔记,记录下如何将 Windows 的磁盘共享给 Linux 存数据使用。
王诗翔呀
2023/02/28
4.9K0
将 windows 磁盘作为共享盘挂载到 Linux 上
C++返回vector/将vector作为参数传递
在C++里很多时候我们会遇到函数想返回两个以上结果的情况,这时候可以用数组(vector)、类来作为容器返回,也可以声明一个全局变量的数组,将数值存放在数组里解决。
vincentbbli
2021/08/18
5.5K0
将 Redis 作为图数据库
1. 简介 Redis 在 4.0 中正式支持了Module模块系统,使其可以进行丰富的扩展 图数据库的应用越来越广泛,RedisGraph 就是一个 Redis Module,可以将 Redis 变为一个高性能的图数据库 图数据库中的2个核心概念:点、边 点 用来描述实体,边 用来描述实体间的关系 实体有多个属性,Redis 中的 Hash 结构就是存储实体的最佳选择,图中的一个节点就是 Redis 中的一个 Hash 例如,假设有2个实体: 一个‘人’的实体描述的是‘奥巴马’,有两个属性,年龄 - 55
dys
2018/04/04
2.5K0
打开迷你云时显示apache2.2端口被占用
今天本地测试phpcms,下载套件后发现Apache2.2无法启动,搜网后发现,估计是80端口被占用,由于之前测试过asp,所以顺便把iis服务也关了,随后用netstat -nao 命令查找80端口是否被占用,发现占用80端口的竟然会是qq旋风!!将其关闭后,Apache2.2顺利启动!步骤如下图: 1、关闭iisAdmin服务,但这个不一定是影响Apache2.2的元凶,但关了保险。关闭方法:依次打开:控制面板>管理工具>服务,找到iisAdmin服务,右击后属性,停止服务,并选择手动启动,下
闵开慧
2018/03/30
1.4K0
wordpress网站提示“建立数据库连接时出错”
wordpress网站提示“建立数据库连接时出错”这个问题用过wordpress的博主应该都会碰到这个问题。前段时间由于升级了数据库跟PHP版本导致wordpress经常提示数据库出错导致整个服务器都会卡死。服务器内的网站全部打不开!在网上也搜集了很多的关于‘wordpress网站提示“建立数据库连接时出错’同样的问题,但是答案各有差异。最终也没能修复好。
青梅煮码
2023/03/13
2.3K0
wordpress网站提示“建立数据库连接时出错”
SQL Server 2008 附加数据库时出错
版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/huyuyang6688/article/details/10079741
DannyHoo
2018/09/13
5.9K0
SQL Server 2008 附加数据库时出错
Erlang 03 - Erlang缺陷
Erlang代码具有较为良好的可读性, 其原因之一就在于语义简明. 大部分情况下, 每个操作的成本都清晰可辨, 没有隐式调用的对象构造函数和析构函数, 没有运算符重载(因此+运算符局部可能偷偷摸摸的复制整个对象), 没有虚函数表带来的间接调用, 没有临界区, 也没有阻塞式的消息发送原语. 当然, 函数调用几乎是“无所不能”的, 他们的行为并不是一目了然, 但通常每个函数都附有清晰的文档.
Reck Zhang
2021/08/11
1.7K0
WordPress 建立数据库连接时出错解决方法
周末外出和朋友一起钓鱼去了,晚上回来准备在自己的米扑博客(http://blog.mimvp.com)写一篇钓鱼游记,打开电脑结果发现博客网站打不开了,提示”建立数据库连接时出错“
阳光岛主
2019/02/18
3.2K0
WordPress 建立数据库连接时出错解决方法
将多说作为静态页面的数据库
前几天想做一个测试,思路就是获取 UA 并保存然后分析数据。就实现 UA 获取功能做一个 HTML 页面使用 JavaScript 就可以实现了,这样正好可以托管到 Github Pages 上,连服务器都省下了。但实现一个数据存放统计功能,单纯获取信息是不行的,我们还需要一个数据库来存放数据,然后读取分析。
Denis
2023/04/15
5280
zblogasp安装时出错,左侧显示无法使用Access数据库
今天突然想起之前的一个网站博客,感觉还不错,但它是zblogasp的,所以想移植到zblogphp版本,但是把网站数据恢复之后登陆后台显示,数据库连接出错,因为asp+access类型,目录位置都对,所以可能是access数据库连接失败。
李洋博客
2021/06/15
4.6K0
如何修复WordPress中的“建立数据库连接时出错”?
"建立数据库连接时出错",这可能是使用WordPress最常见错误之一,所有使用WordPress建站的用户都可能看到过此消息。不用担心,这是一个非常普遍的问题,有很多解决方法。
晓得博客
2021/05/17
5.3K0
Windows端口被占用_windows如何打开端口
【Win + R】:使用快捷键打开“运行”窗口 输入【cmd】,点击确定,打开“命令”窗口
全栈程序员站长
2022/10/04
4.4K0
Windows端口被占用_windows如何打开端口
打开redis远程访问端口_linux端口开放命令
最近我在阿里云ESC上购买了一台服务器,但是在安装完redis后,我在本地的电脑上怎么也没法调用这台服务器上面的redis服务。
全栈程序员站长
2022/09/29
6K0
Erlang连接&操作mysql数据库
地址:https://github.com/dizzyd/erlang-mysql-driver
山海散人
2021/03/03
1.1K0
点击加载更多

相似问题

高效的Erlang端口驱动程序

10

打开端口时出错

13

打开Python串行端口时出错

20

将Erlang端口示例转换为Erlang

15

C++打开与CreateFile的共享串行端口

10
添加站长 进交流群

领取专属 10元无门槛券

AI混元助手 在线答疑

扫码加入开发者社群
关注 腾讯云开发者公众号

洞察 腾讯核心技术

剖析业界实践案例

扫码关注腾讯云开发者公众号
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文