C++ 和 Java 进程之间的共享内存

Posted

技术标签:

【中文标题】C++ 和 Java 进程之间的共享内存【英文标题】:Shared memory between C++ and Java processes 【发布时间】:2013-03-03 03:00:48 【问题描述】:

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

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

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

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

迷路了。

非常感谢。

【问题讨论】:

【参考方案1】:

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

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

【讨论】:

你有 android 上使用 0MQ 的第一手经验吗?【参考方案2】:

如果您有共享内存,例如使用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);

【讨论】:

@cam79 请将此标记为答案。经过测试,它可以工作。

以上是关于C++ 和 Java 进程之间的共享内存的主要内容,如果未能解决你的问题,请参考以下文章

用于导入的“RegExp.tlb”的 C++ COM 接口是不是在进程之间隐式共享内存?

在 C++ 中父进程和子进程之间共享队列

共享内存

在 c++ 程序中有效使用 POSIX 共享内存

Gunicorn 在多处理进程和工作进程之间共享内存

如何通过子进程之间的共享内存共享信号量?