JNI Calling API - NoClassDefFoundError (C / Java)

I'm trying to get my feet wet with JNI because I have a C application that needs to access one function of the Java library (no C-equivalent library). I wrote a very simple test program to load a Java virtual machine from C and call a static function and get the return value.

Unfortunately, I cannot get the class to load correctly. Although this will probably work, I think my ClassPath is correct: when I use a command javawith the same ClassPath in the same directory, the class loads and runs fine.

Environment:
Ubuntu 8.04
Java JRE & SDK 1.6
GCC Server

My current working directory is always /home/me/project.

Here is what I get when I run the java( java -Djava.class.path=/home/me/project/ -verbose my.ClassABC) command :

[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]

Here is what I get when I run my C ( ./myClassABC) program :

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

Here is my gcc command line:

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

My C ( myClassABC.c) code :

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

And my java code is only for # $% @s and giggles (compiles to /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;
    }
}

Thanks
Brian

+5
source share
1 answer

Just replace it

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

from

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

and everything should be fine.

Try adding again.

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

All Articles