最近在学习 JNI 来执行 C 代码。当然,我做了网络上的基本示例。现在我正在尝试加载一个可以进行动态库加载(dlopen)的 C 库。但我正在与错误作斗争。我发布了我的 Java 代码、C++ 代码和错误。
我的 Java 类
/**
*
* @author glassfish
*/
public class MediationJniWeb {
public String library ;
static {
System.loadLibrary("-core-web");
}
/**
*
* @param library name of mediation core library [32]
* @param method name of method to be executed [128]
* @param parameters parameters of method [10240]
* [partype1,value1,...,valuen]...[partypen,value1,..,valuen]
* @return
*/
private native String execute();
public static void main(String args[]) {
//new MediationJniWeb().callToFunction(null, null, null) ;
MediationJniWeb jniWeb = new MediationJniWeb();
jniWeb.library = "libtest.so" ;
System.out.println(jniWeb.execute());
}
}
我生成 .class 文件
javac MediationJniWeb
我生成 .h 文件
javah -jni MediationJniWeb
我的 MediationJniWeb.h 文件是
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class MediationJniWeb */
#ifndef _Included_MediationJniWeb
#define _Included_MediationJniWeb
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: MediationJniWeb
* Method: execute
* Signature: ()Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_MediationJniWeb_execute
(JNIEnv *, jobject);
#ifdef __cplusplus
}
#endif
#endif
和我的 MediationJniWEB.cpp 文件
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
#include <dlfcn.h>
#include <iostream>
#include "MediationJniWeb.h"
using namespace std ;
typedef void (*test)(string);
/*
* Class: MediationJniWeb
* Method: execute
* Signature: ()Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_MediationJniWeb_execute
(JNIEnv * env, jobject obj){
const char * str_library ;
jfieldID fid_library ;
jstring jstr_library ;
jboolean isCopy ;
jclass cls = env->GetObjectClass( obj) ;
fid_library = env->GetFieldID( cls,"library", "Ljava/lang/String;");
jstr_library = ( jstring )env->GetObjectField( obj,fid_library);
str_library = env->GetStringUTFChars( jstr_library, &isCopy) ;
void* handle = dlopen(str_library, RTLD_NOW); // open libtest.so
if ( 0 == handle ) {
cout << "failed to load 'libtest.so'. " << dlerror () <<endl;
exit(1);
}
test t = (test)dlsym(handle, "_Z8testfuncSs"); // run default method
if ( 0 == t ) {
cout << "failed to load 'testfunc()'." << endl;
exit(1);
}
t("Hello, World!");
dlclose(handle);
/*
*/
return env->NewStringUTF( str_library); // I just return library name just for fun
}
}
我编译
g++ -shared -fpic -I//include/ -I//include/linux/ MediationJniWeb.cpp -o lib-core-web.so MediationJniWeb.cpp -ldl
这会生成 lib-core-web.so 文件。然后我将它复制到 $HOME/lib 并配置
LD_LIBRARY_PATH=$HOME/lib
现在我创建了我的库 libtest.so,它将由 lib-core-web.so 执行
我为共享库 mylib.cpp 创建文件
#include <iostream>
#include <string>
using namespace std;
void testfunc(string s)
{
cout << s << endl;
}
我编译这个将作为一个共享库工作
g++ -shared -fpic -o libtest.so mylib.cpp
此命令生成 libtest.so 文件.. 而且,我将其复制到 $HOME/lib
这就是我从 JNI 调用 C++ 库以加载动态库的全部工作。当我执行 MediationJniWeb java 类时出现此错误
加载失败。libtest.so:无法打开共享对象文件:没有这样的文件或目录
我与 libtest.so 有什么关系?我必须把它放在哪里?
我记得只用正确的路径配置 LD_LIBRARY_PATH 变量,JVM 应该知道在哪里可以找到所有需要加载的库。
请帮助您发表评论,让我知道我的错误在哪里。
提前致谢 !