13

当我从 Java 调用 C/C++ 时,JavaVM 或 JNI 是否创建了一个新线程来在我的 Java 线程等待时运行 C/C++ 代码?我问这个是因为我的 C/C++ 代码在 GPU 上运行一些东西,我需要检查一个特定的缓冲区来获取结果。得到结果后,我需要再次调用我的 Java 函数。

所以我想在 C++ 端创建一个线程,不断检查缓冲区,一旦有一些数据可用,就回调 Java 端。

4

1 回答 1

19

JNI 不会在后台创建任何新线程。本机函数与调用本机函数的 java 方法在同一线程中执行。反之亦然,当本机代码调用 java 方法时,java 方法与调用该方法的本机代码在同一线程中执行。

它有后果 - 当本机函数返回时,本机函数调用返回到 java 代码,而当被调用的 java 方法返回时,本机代码继续执行。

当本机代码执行应在单独线程中运行的处理时,必须显式创建线程。您可以创建一个新的 java 线程并从该专用线程调用本机方法。或者您可以在本机代码中创建一个新的本机线程,启动它并从本机函数返回。

// Call a native function in a dedicated java thread
native void cFunction();
...
new Thread() {
    public void run() {
        cFunction();
    }
};

// Create a native thread - java part
native void cFunction()
...
cFunction();

//  Create a native thread - C part
void *processing_function(void *p);
JNIEXPORT void JNICALL Java____cFunction(JNIEnv *e, jobject obj) {
    pthread_t t;
    pthread_create(&t, NULL, processing_function, NULL);    
}

如果您使用第二种变体并且您想从本地创建的线程调用 java 回调,您必须将该线程附加到 JVM。怎么做?请参阅JNI Attach/Detach 线程内存管理...

于 2016-07-17T17:14:53.473 回答