使用sun.misc.Unsafe获取Java数组项的地址?

Jul*_*les 7 java memory pointers unsafe native

我正在努力理解sun.misc.Unsafe的文档 - 我想因为它不适合一般用途,没有人真的为让它可读而烦恼 - 但实际上我真的需要一种方法来找到一个元素的地址一个数组(以便我可以将指针传递给本机代码).有没有人有任何工作代码这样做?它可靠吗?

Woj*_*zyk 6

这是一个工作样本.但请注意,因为您可能会因为不正确的Unsafe类使用而轻易崩溃JVM .

import java.lang.reflect.Field;

import sun.misc.Unsafe;

public class UnsafeTest {

    public static void main(String... args) {
        Unsafe unsafe = null;

        try {
            Field field = sun.misc.Unsafe.class.getDeclaredField("theUnsafe");
            field.setAccessible(true);
            unsafe = (sun.misc.Unsafe) field.get(null);
        } catch (Exception e) {
            throw new AssertionError(e);
        }

        int ten = 10;
        byte size = 1;
        long mem = unsafe.allocateMemory(size);
        unsafe.putAddress(mem, ten);
        long readValue = unsafe.getAddress(mem);
        System.out.println("Val: " + readValue);

    }
}
Run Code Online (Sandbox Code Playgroud)

  • @StephenC,代码非常好,它基本上是`char*mem = malloc(size); ...`它从未被GC触及并导致本机C泄漏,除非重新获得. (4认同)
  • 该代码必须完全安全,因为它正是DirectByteBuffer的工作原理.不幸的是,它没有做我想要的,即访问现有数组中的数据. (2认同)
  • 代码很好,但您还需要考虑释放分配的内存.请参阅DirectByteBuffer以了解其中一种方法.还要考虑分配的内存不归零,因此您不能假设您的直接数组初始化为零. (2认同)

Pet*_*rey 6

您可以使用ByteBuffer.allocateDirect()直接缓冲区而不是使用数组.它具有字段中的地址,并且该地址在ByteBuffer的生命周期内不会改变.直接ByteBuffer使用最小的堆空间.您可以使用反射获取地址.


您可以使用Unsafe获取地址,问题是GC可以随时移动它.对象未在内存中修复.

在JNI中,您可以使用特殊方法将数据复制到Java对象或从Java对象复制数据以避免此问题(以及其他问题)如果您希望在具有C代码的对象之间交换数据,我建议您使用这些方法.