我究竟如何从Java调用本机静态库?我正在使用Java 8。
在我看来,我应该能够在带有嵌入式JVM的C ++程序中定义JNI_OnLoad_library,但是当我调用System.loadLibrary时,我的VM仍然会死。以下仅打印“Hello from main”。
main.cpp中:
#include <jni.h>
#include <iostream>
extern "C" {
JNIEXPORT jint JNI_OnLoad_hello(JavaVM *vm, void *reserved) {
std::cout << "Hello World" << std::endl;
return JNI_VERSION_1_8;
}
}
int main(int argc, char** argv) {
JavaVM *jvm;
JNIEnv *env;
JavaVMInitArgs vm_args;
jint error;
JavaVMOption* options = new JavaVMOption[argc-1];
for(int i = 0;i < argc-1;i++) {
options[i].optionString = argv[i+1];
}
vm_args.nOptions = argc-1;
vm_args.options = options;
vm_args.version = JNI_VERSION_1_8;
JNI_CreateJavaVM(&jvm, (void**)&env, &vm_args);
jclass cls = env->FindClass("Main");
jclass stringCls = env->FindClass("java/lang/String");
jmethodID mid = env->GetStaticMethodID(cls, "main", "([Ljava/lang/String;)V");
jobjectArray mainArgs = env->NewObjectArray(0, stringCls, NULL);
env->CallStaticVoidMethod(cls, mid, mainArgs);
jvm->DestroyJavaVM();
}
Main.java:
class Main {
public static void main(String[] args) {
try {
System.out.println("Hello from main");
System.loadLibrary("hello");
System.out.println("Hello after main");
} catch (Exception e) {
e.printStackTrace();
}
}
}
答案 0 :(得分:0)
我终于能够通过将--export-dynamic传递给链接器来让JVM找到JNI_OnLoad_hello。这允许共享库通过在可执行文件中查找符号来解析符号。另外,System.loadLibrary抛出了我所怀疑的UnsatisfiedLikeErrors,但由于某种原因,我必须使用CheckException和DescribeException在JNI中处理它们。