首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >问答首页 >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

复制
相关文章
触屏事件
TouchEvent是一类描述手指在触摸平面(触摸屏。触摸板等)的状态变化的事件,这类事件用于描述一个或多个触点,使开发者可以检测触点的移动,触点的增加和减少等等
切图仔
2022/09/08
7720
APP技巧:安卓原生跨屏协同终于来了,还能和 Windows「隔空投送」
谷歌在 CES 2022 上宣布了至少 13 种不同的新软件功能,从类似 AirPods 的快速切换到此前承诺过的软件,可以在 Chromebook 上镜像你的 Android 文本应用程序等。这是谷歌称为“Better Together”的计划的一部分,外界认为谷歌是想追赶苹果生态系统,学苹果一样搞整合。
IT技术分享社区
2022/02/17
2.4K0
APP技巧:安卓原生跨屏协同终于来了,还能和 Windows「隔空投送」
触屏事件
移动端浏览器兼容性较好,我们不需要考虑以前 JS 的兼容性问题,可以放心的使用原生 JS 书写效果,但是移动端也有自己独特的地方。比如触屏事件 touch(也称触摸事件),Android和 IOS 都有。
清出于兰
2020/10/26
7930
触屏事件
触屏事件
移动端浏览器兼容性较好,我们不需要考虑以前 JS 的兼容性问题,可以放心的使用原生 JS 书写效果,但是移动端也有自己独特的地方。比如触屏事件 touch(也称触摸事件),Android和 IOS 都有。
梨涡浅笑
2022/05/08
7390
触屏事件
关于选购笔记本电脑
与电脑打交道十多年来,以及从事程序数年转网络安全三年来,在与985空间安全研究生、电脑经销商,网络安全实验室负责人、讨论及对购买电脑的理解,写下此文。
红客突击队
2022/09/29
4.2K0
关于选购笔记本电脑
JavaScript——触屏事件
移动端浏览器兼容性较好,我们不需要考虑以前js的兼容性问题,可以放心的使用原生js书写效果,但是移动端也有自己独特的地方。比如触屏事件touch(也称触摸事件),Android和IOS都有。
岳泽以
2022/10/26
2.1K0
JavaScript——触屏事件
IOS UIResponder 触屏
class ViewController:UIViewController { override func viewDidLoad() { super.viewDidLoad() } override func touchesBegan(_ touches:Set<UITouch>,with event:UIEvent?) { print(“touchesBegan”); } override func touchesMoved(_ touches:Set<UITouch>, with event:UIEvent?) { print(“touchesMoved”); } override func touchesEnded(_ touches:Set<UITouch>,with event:UIEvent?){ print(“touchesEnded”); } override func touchesCancelled(_ touches:Set<UITouch>, with event:UIEvent?) { print(“touchesCancelled”); } }
用户5760343
2019/07/08
1.4K0
IOS UIResponder 触屏
Android触屏事件流[通俗易懂]
其中ACTION_DOWN和ACTION_UP在一次触屏事件中只会触发一次,ACTION_MOVE可能触发任意次(包括0次)。
全栈程序员站长
2022/09/30
1.2K0
android 触屏事件总结
如果view的down事件返回true,则接下去的move,up,cancel,事件最多传递到这个view,不会传递给view的子view
全栈程序员站长
2022/09/30
9040
移动端触屏事件
移动端浏览器兼容性较好,我们不需要考虑以前 JS 的兼容性问题,可以放心的使用原生 JS 书写效果,但是移动端也有自己独特的地方。比如触屏事件 touch(也称触摸事件),Android和 IOS 都有。
星辰_大海
2020/10/09
2.9K0
移动端触屏事件
触屏touchstart 与 click
设计效果:当手指点击或触摸红框线menuList之外的部分时,弹框menuList消失。 问题:在优化触屏版的时候发现如图问题,当menuList弹出,手指触摸屏幕向下滑动时,menuList弹框不消失,只有手指点击menuList之外的部分才消失。 查看代码发现,源代码只定义了click事件: $(doc.body).on('click',function(e) { if (e.target.id != 'menu') headerMenu.hide(); }); 说明触屏版对cl
smy
2018/04/03
2.8K0
触屏touchstart 与 click
界面劫持之触屏劫持
移动智能终端设备由于体积限制,一般都没有鼠标、键盘这些输入设备,用户更多的操作是依靠手指在触屏上的点击或滑动等动作完成。在移动设备上,类似点击劫持的攻击模式,实现了对用户触摸屏操作的劫持攻击,即界面操作劫持攻击的又一种形式——触屏劫持。
第59号实验室
2023/07/04
3330
界面劫持之触屏劫持
谷歌发布手势识别技术,有望实现隔空触屏【智能快讯】
AI 无处不在的时代,每天都有新的技术与研究成果出现。无论学术界还是商界,技术还是产品,AI 的新发现都源源不断,在带给我们全新视角的同时,也引起我们更深的思考。
HyperAI超神经
2019/11/29
7970
谷歌发布手势识别技术,有望实现隔空触屏【智能快讯】
欢迎来到联想智能大屏的新世界
刘嘉对产品团队的持续挑战终于激起了一些轻微的反抗。产品团队“建议”他先试用两天之后再谈问题。
IT创事记
2022/06/24
3680
欢迎来到联想智能大屏的新世界
升级版IPS全贴合触屏相比传统TN+触屏的优势在哪里?
为了简化屏和板的组装过程,提高生产效率,在原先TN+触屏的基础上,推出了升级版的IPS全贴合触屏,弥补了传统触屏组装麻烦、延展性弱、成本高等缺点。
启明智显
2022/06/07
2.1K0
升级版IPS全贴合触屏相比传统TN+触屏的优势在哪里?
Appium常用操作之「微信滑屏、触屏操作」
不需要看到欢迎页面,直接做里面的后续操作就好了。也就是我想记住一些用户的习惯,不要像第一次访问一样,有一个重置与否,给它关闭了。
清菡
2020/12/02
2.1K0
Appium常用操作之「微信滑屏、触屏操作」
Appium常用操作之「微信滑屏、触屏操作」
不需要看到欢迎页面,直接做里面的后续操作就好了。也就是我想记住一些用户的习惯, 不要像第一次访问一样,有一个重置与否,给它关闭了。
清菡
2020/11/04
1.6K0
《中国医生》里的ThinkPad,中小企业智慧化的“引路人”
3200万观影人次、超过12亿元的票房和排名第一的上座率,让《中国医生》成了2021年暑假档最动人的电影。
Alter聊科技
2023/01/12
2540
聚焦应用场景,联想ThinkPad L用11款配置细腻覆盖大客户
在3月26日的联想ThinkPad L系列产品品鉴会上,王立平宣布了一个新的计划,要让ThinkPad L系列占据中国商用大客户市场10%的市场份额——2018年,ThinkPad L系列向商用大客户市场发布了4款、共11个配置版本的新品——这也让ThinkPad L系列成了产品家族中,细分度最高的一个产品系列。
IT创事记
2022/06/16
7840
聚焦应用场景,联想ThinkPad L用11款配置细腻覆盖大客户
一点chromium触屏相关堆栈
       chrome.dll!ui::TouchDispositionGestureFilter::CancelTapIfNecessary 行 394    C++    已加载符号。 >    chrome.dll!ui::TouchDispositionGestureFilter::SendGesture 行 364    C++    已加载符号。      chrome.dll!ui::TouchDispositionGestureFilter::FilterAndSendPacket 行 302    C++    已加载符号。      chrome.dll!ui::TouchDispositionGestureFilter::SendAckedEvents 行 254    C++    已加载符号。      chrome.dll!ui::TouchDispositionGestureFilter::OnTouchEventAck 行 218    C++    已加载符号。      chrome.dll!content::TouchEmulator::HandleTouchEventAck 行 327    C++    已加载符号。      chrome.dll!content::TouchEventAckQueue::ProcessAckedTouchEvents 行 196    C++    已加载符号。      chrome.dll!content::TouchEventAckQueue::MarkAcked 行 182    C++    已加载符号。      chrome.dll!content::RenderWidgetHostImpl::OnTouchEventAck 行 2802    C++    已加载符号。      chrome.dll!content::InputRouterImpl::OnTouchEventAck 行 419    C++    已加载符号。      chrome.dll!content::PassthroughTouchEventQueue::AckCompletedEvents 行 241    C++    已加载符号。      chrome.dll!content::PassthroughTouchEventQueue::ProcessTouchAck 行 154    C++    已加载符号。      chrome.dll!content::InputRouterImpl::TouchEventHandled 行 634    C++    已加载符号。      chrome.dll!base::internal::Invoker<...>::RunOnce 行 645    C++    已加载符号。      chrome.dll!content::mojom::WidgetInputHandler_DispatchEvent_ForwardToCallback::Accept 行 1983    C++    已加载符号。      chrome.dll!mojo::InterfaceEndpointClient::HandleValidatedMessage 行 428    C++    已加载符号。      chrome.dll!mojo::internal::MultiplexRouter::ProcessIncomingMessage 行 874    C++    已加载符号。      chrome.dll!mojo::internal::MultiplexRouter::Accept 行 598    C++    已加载符号。      chrome.dll!mojo::Connector::DispatchMessageW 行 524    C++    已加载符号。      chrome.dll!mojo::Connector::ReadAllAvailableMessages 行 600    C++    已加载符号。      chrome.dll!mojo::Connector::OnHandleReadyInternal 行 437    C++    已加载符号。      chrome.dll!base::internal::Invoker<...>::RunOnce 行 645    C++    已加载符号。
龙泉寺扫地僧
2020/06/12
1.5K0

相似问题

如何在联想ThinkPad T400上安装Linux

50

触摸Linux上的触屏中断输入

10

联想Thinkpad Dockingstation混合C-USB不识别外部监视器

20

联想ThinkPad静音LED停止工作

10

未在联想Thinkpad上工作的鼠标间隔

10
添加站长 进交流群

领取专属 10元无门槛券

AI混元助手 在线答疑

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

洞察 腾讯核心技术

剖析业界实践案例

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