2009-11-23 6 views
5

J'essaie de me mouiller les pieds avec JNI parce que j'ai une application en C qui a besoin d'accéder à une seule fonction de bibliothèque Java (pas de bibliothèque C-équivalente). J'ai écrit un programme de test très simple pour charger une machine virtuelle Java à partir de C et appeler une fonction statique pour obtenir la valeur de retour.API Invocation JNI - NoClassDefFoundError (C/Java)

Malheureusement, je n'arrive pas à charger la classe correctement. Bien que cela se résume probablement à cela, je pense que mon ClassPath est correct: lorsque j'utilise la commande java avec le même ClassPath dans le même répertoire, la classe se charge et s'exécute parfaitement.

Environnement:
Ubuntu 8.04 serveur
Java SDK 1.6 JRE &
gcc

Mon présent répertoire de travail est toujours /home/me/project.

Voici ce que je reçois quand je lance la commande 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] 

Voici ce que je reçois quand je lance mon programme (./myClassABC) C:

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

Voici ma ligne de commande 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

Mon code 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.] 
} 

Et mon code java, juste pour # $% @ s et fou rire (se compile à /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; 
    } 
} 

Merci,
Brian

Répondre

10

Il suffit de remplacer cette

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

avec

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

et vous devriez être bien.

essayer autre ajouter

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

homme Oh, merci beaucoup! En revenant sur la spécification et les choses que je vois maintenant que j'ai regardé par-dessus. – HalfBrian

Questions connexes