2009-11-23 16 views
5

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

Respuesta

10

basta con sustituir este

cls = (*env)->FindClass(env,"my.ClassABC"); 

con

cls = (*env)->FindClass(env,"my/ClassABC"); 

y que debe estar bien.

Else probar a añadir

... 
JavaVMOption options[3]; 
... 
options[2].optionString = "-verbose:jni"; 
... 
+0

Oh hombre, muchas gracias! Mirando hacia atrás en la especificación y las cosas que ahora veo que revisé eso. – HalfBrian