如何使用 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());
}
}
我想使用巴拿马项目的外部内存访问 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());
}
}