决定使用JNI,实际是为了能够将通讯与调用后台的lucene索引,因此老大决定要采用这种方式来实现index的多机分布式的索引服务。接到任务,使用C++来调用Lucene的java查询的封装类。
//创建JVM
JNIEnv* env=NULL;
JavaVM* jvm=NULL;
bool Object::BeginJVM()
...{
JavaVMOption options[4];
JavaVMInitArgs vm_args;
//各种参数
options[0].optionString="-Xmx512m";
options[1].optionString="-Xms128m";
//大家注意这里为你需要使用的java类class所在的目录
options[2].optionString="-Djava.class.path=.:../lucene-core-2.0.0.jar";
options[3].optionString="-Djava.compiler=NONE";
//该地方我也不太清除,我的JDK版本是1.5.0,但好像没有JNI_VERSION_1_5
vm_args.version=JNI_VERSION_1_4;
vm_args.options=options;
vm_args.nOptions=4;
//创建JVM,获得jvm和env
int res = JNI_CreateJavaVM(&jvm,(void **)&env, &vm_args);
if(res == JNI_ERR)
...{
printf("Error invoking the JVM");
return false;
}
return true;
}
bool Object::EndJVM()
...{
//关闭JVM
jvm->DestroyJavaVM();
return true;
}
注意编译的时候,要加上-I指定头文件目录,-L指定libjvm.so的所在目录,如我的:
g++ -I /usr/java/jdk1.5.0/include/ -I /usr/java/jdk1.5.0/include/linux/ -L /usr/java/jdk1.5.0/jre/lib/i386/server/ -o startJVM startJVM.cpp -ljvm这样至少可以以C/C++的方式启动了java的JVM,下篇将是实际读取,调用java中的数据和接口。
对了有些可能会碰到的出错问题:第一,libjvm.so :no such file or directory. 第二,就是N多的undefine ....的
总结下来,第一种是因为没有加入-L 指明lib所在路径,并加上-ljvm ;第二种是由于没有加入-I 指明头文件和linux下目录的头文件。
另外,在运行的时候,可能会报错cannot find shared lib:libjvm.so,问题是没有加入run_lib_directory,也就是说要设置LD_LIBRARY_PATH。设置的命令为:export LD_LIBRARY_PATH=/usr/java/jdk1.5.0/jre/lib/i386/server
本文分享了作者在Linux环境下使用JNI实现C++调用Java Lucene索引服务的经验,详细介绍了启动Java虚拟机并设置必要参数的过程。
1213





