1

我想使用 Project 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(例如对象)中?如果是这种情况,当 VarHandles 仅支持原始载体时,如何使用 VarHandle 取消引用该 MemorySegment?

4

1 回答 1

1

有没有办法将非基元存储在 MemorySegment(例如对象)中?

不,至少不是直接的。对象由 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());
    }
}
于 2021-06-11T21:58:29.017 回答