2009-11-23 13 views
5

Sto cercando di bagnarmi i piedi con JNI perché ho un'applicazione in C che deve accedere a una singola funzione di libreria Java (nessuna libreria equivalente C). Ho scritto un programma di test molto semplice per caricare una VM Java da C e chiamare una funzione statica e ottenere il valore di ritorno.API di richiamo JNI - NoClassDefFoundError (C/Java)

Sfortunatamente, non riesco a caricare correttamente la classe. Anche se probabilmente si ridurrà ad esso, penso che il mio ClassPath sia corretto: quando uso il comando java con lo stesso ClassPath nella stessa directory, la classe si carica ed esegue perfettamente.

Ambiente:
Ubuntu 8.04 Server
Java JRE & SDK 1.6
gcc

mia directory di lavoro attuale è sempre /home/me/project.

Ecco che cosa ottengo quando faccio funzionare l'ordine 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] 

Ecco che cosa ottengo quando eseguo il mio programma in C (./myClassABC):

[Loaded ...] 
[Loaded my.ClassABC from file:/home/me/project/] 
Exception in thread "main" java.lang.NoClassDefFoundError: my.ClassABC 
Failed to get class 

Ecco la mia linea di 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

Il mio codice 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.] 
} 

E il mio codice Java, solo per # $% @ s e risatine (viene compilato 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; 
    } 
} 

Grazie,
Brian

risposta

10

basta sostituire questo

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

con

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

e si dovrebbe andare bene.

Else provare aggiungendo

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

Oh uomo, grazie mille! Guardando indietro alle specifiche e alle cose che ora vedo, ho guardato oltre. – HalfBrian