Estoy tratando de mojarme los pies con JNI porque tengo una aplicación en C que necesita acceder a una sola función de biblioteca Java (no hay una biblioteca equivalente en C). He escrito un programa de prueba muy simple para cargar una máquina virtual Java desde C y llamar a una función estática y obtener el valor de retorno.JNI Invocation API - NoClassDefFoundError (C/Java)
Lamentablemente, no puedo hacer que la clase cargue correctamente. Aunque probablemente se reducirá a eso, creo que mi ClassPath es correcta: cuando uso el comando java
con el mismo ClassPath en el mismo directorio, la clase se carga y se ejecuta perfectamente.
Medio Ambiente:
Ubuntu 8.04 servidor
Java JRE & SDK 1.6
gcc
Mi directorio de trabajo actual es siempre /home/me/project
.
Esto es lo que me pasa cuando funciono con el comando java
(java -Djava.class.path=/home/me/project/ -verbose my.ClassABC
):
[Loaded ...] (many loads)
[Loaded my.ClassABC from file:/home/me/project/]
Hello test
[Loaded java.lang.Shutdown from shared objects file]
[Loaded java.lang.Shutdown$Lock from shared objects file]
Aquí es lo que me pasa cuando corro mi programa C (./myClassABC
):
[Loaded ...]
[Loaded my.ClassABC from file:/home/me/project/]
Exception in thread "main" java.lang.NoClassDefFoundError: my.ClassABC
Failed to get class
Aquí está mi línea de comando gcc:
gcc -o myClassABC myClassABC.c -I/usr/lib/jvm/java-6-sun-1.6.0.16/include/ -I/usr/lib/jvm/java-6-sun-1.6.0.16/include/linux -L/usr/lib/jvm/java-6-sun-1.6.0.16/jre/lib/i386/server/ -ljvm
Mi código C (myClassABC.c
):
int main(int argc,char **argv)
{
JNIEnv *env;
JavaVM *jvm;
jint res;
jclass cls;
jmethodID mid;
jstring jstr;
jclass stringClass;
jobjectArray args;
JavaVMInitArgs vm_args;
JavaVMOption options[2];
options[0].optionString =
"-Djava.class.path=."; // or "-Djava.class.path=/home/me/project/";
options[1].optionString =
"-verbose";
vm_args.version = JNI_VERSION_1_6;
vm_args.options = options;
vm_args.nOptions = 2;
vm_args.ignoreUnrecognized = JNI_FALSE;
/* Create the Java VM */
res = JNI_CreateJavaVM(&jvm, (void**)&env, &vm_args);
if (res < 0) {
fprintf(stderr, "Can't create Java VM\n");
exit(1);
}
if ((*env)->ExceptionOccurred(env)) {
(*env)->ExceptionDescribe(env);
}
cls = (*env)->FindClass(env,"my.ClassABC");
if (cls == NULL) {
if ((*env)->ExceptionOccurred(env)) {
(*env)->ExceptionDescribe(env);
}
printf("Failed to get class\n");
exit(1);
}
[call methods, etc.]
}
Y mi código java, sólo por # $% @ s y la risa (se compila a /home/me/project/my/ClassABC.class
):
package my;
class ClassABC {
public static void main(String[] args) {
System.out.println(ClassABC.getPassword("test"));
return;
}
static String getPassword(String filename)
{
return "Hello "+filename;
}
}
Gracias,
Brian
Oh hombre, muchas gracias! Mirando hacia atrás en la especificación y las cosas que ahora veo que revisé eso. – HalfBrian