如何使用 Java Panama 在本机内存中存储对象数组

How to store an Object-Array in native Memory using Java Panama

我想使用巴拿马项目的外部内存访问 API 在本机内存中实现数据结构。

为此,我需要一个底层对象数组 (Object[]) 作为条目。

在外部内存访问的所有示例中 API,MemorySegments 仅用于存储和检索基元,如下所示:

//I would like to use Object here instead of int
ResourceScope scope = ResourceScope.newConfinedScope();
SequenceLayout layout = MemoryLayout.sequenceLayout(100, MemoryLayouts.JAVA_INT); 
VarHandle intHandle = seq.varHandle(int.class, sequenceElement());
MemorySegment segment = MemorySegment.allocateNative(layout, scope);

有没有办法将非基元存储在 MemorySegment(例如对象)中? 如果是这种情况,当 VarHandle 仅支持原始载体时,我如何使用 VarHandle 取消引用该 MemorySegment?

Is there a way to store non primitives in a MemorySegment (e.g. Object)?

不,至少不是直接的。对象由 Java 运行时管理,它们不能安全地存储在本机内存中(例如,因为垃圾收集器无法跟踪本机内存中对象内部的对象引用)。​​

但是,如评论中所述,出于您的目的,将数据存储在本机内存中的对象内可能就足够了。例如,如果一个对象只包含原始字段(尽管可以对对象字段递归执行相同的操作),则可以将每个此类字段分别写入本机内存。例如(使用 JDK 16 API):

public static void main(String[] args) {
    try (MemorySegment segment = MemorySegment.allocateNative(Widget.NATIVE_LAYOUT)) {
        Widget widget1 = new Widget(1, 2);
        widget1.write(segment);

        Widget widget2 = Widget.read(segment);
        System.out.println(widget2); // Widget[x=1, y=2]
    }
}

record Widget(int x, int y) {
    static final MemoryLayout NATIVE_LAYOUT = MemoryLayout.ofStruct(
        MemoryLayouts.JAVA_INT.withName("x"),
        MemoryLayouts.JAVA_INT.withName("y")
    );

    static final VarHandle VH_x = NATIVE_LAYOUT.varHandle(int.class, MemoryLayout.PathElement.groupElement("x"));
    static final VarHandle VH_y = NATIVE_LAYOUT.varHandle(int.class, MemoryLayout.PathElement.groupElement("y"));

    public static Widget read(MemorySegment segment) {
        int x = (int) VH_x.get(segment);
        int y = (int) VH_y.get(segment);
        return new Widget(x, y);
    }

    public void write(MemorySegment segment) {
        VH_x.set(segment, x());
        VH_y.set(segment, y());
    }
}