24

我的目标是将数据从 C++ 进程传递到 Java 进程,然后接收返回的结果。

我已经通过命名管道实现了这一点,但我更愿意共享数据而不是传递或复制它,假设访问会更快。

最初,我想用 C++ 创建一个可以用 Java 读写的共享段,但我不确定这是否可以通过 JNI 实现,更不用说安全了。

我相信在 Java 中可以使用 ByteBuffer.allocateDirect 分配内存,然后使用 GetDirectBufferAddress 访问 C++ 中的地址,但如果我是正确的,这是针对 JNI 中的本机调用,我无法在我的 C++ 进程中获取此地址?

丢失的。

提前谢谢了。

4

2 回答 2

17

如果您有共享内存,例如使用CreateFileMapping(Windows) 或shmget(Unix),您所需要的只是 Java 端的本机方法。然后您可以创建一个ByteBuffer直接访问共享内存的方法,NewDirectByteBuffer如下所示:

JNIEXPORT jobject JNICALL Java_getSharedBuffer(JNIEnv* env, jobject caller) {
    void* myBuffer;
    int bufferLength;

现在你必须得到一个指向共享内存的指针。在 Windows 上,你会使用这样的东西:

    bufferLength = 1024; // assuming your buffer is 1024 bytes big
    HANDLE mem = OpenFileMapping(FILE_MAP_READ, // assuming you only want to read
           false, "MyBuffer"); // assuming your file mapping is called "MyBuffer"
    myBuffer = MapViewOfFile(mem, FILE_MAP_READ, 0, 0, 0);
    // don't forget to do UnmapViewOfFile when you're finished

现在你可以创建一个ByteBuffer由这个共享内存支持的:

    // put it into a ByteBuffer so the java code can use it
    return env->NewDirectByteBuffer(myBuffer, bufferLength);
}
于 2013-03-14T17:20:46.517 回答
4

您是否考虑过使用0MQ,它同时支持JavaC++,并且会更可靠。我认为如果你想在 Java 中做共享内存,它必须通过 JNI,上次我看没有其他方法可以做到这一点。

这表明如果你走那条路,你必须通过JNI来做。虽然我找到的解决方案是特定于 Windows 的,但可能不适用于您。

于 2013-03-14T16:26:32.497 回答