Создать класс JAVA из кода C

У меня есть класс в Java называется XMLDOMDocument у этого класса есть несколько методов CreateXML, AddNode, RemoveNode и т. д. Я вызываю эти методы из кода C с помощью Java Native Interface, но для этого я затем перехожу к объекту кода C моего класса JAVA, вызывая env->GetObjectClass(myclass) Я получаю свой класс от этого объекта и вызываю методы класса JAVA.

Я хочу знать, могу ли я делать то же самое (вызывать методы JAVA), не передавая мой класс в C-код. Могу ли я создать класс JAVA прямо в коде C, а затем вызвать его методы.

отредактированный

И если у меня есть JavaVM в моем коде C, я могу создать новый экземпляр класса JAVA в коде C с помощью этой Java VM.

отредактированный

Я думаю, что нашел что-то полезное Создание виртуальной машины Java, но я хочу понять, какое значение должно быть указано #define USER_CLASSPATH? Если это должно быть имя пакета com.fido.android.framework.service

public class Prog {
     public static void main(String[] args) {
          System.out.println("Hello World " + args[0]);
     }
 }

#include <jni.h>

 #define PATH_SEPARATOR ';' /* define it to be ':' on Solaris */
 #define USER_CLASSPATH "." /* where Prog.class is */

 main() {
     JNIEnv *env;
     JavaVM *jvm;
     jint res;
     jclass cls;
     jmethodID mid;
     jstring jstr;
     jclass stringClass;
     jobjectArray args;

 #ifdef JNI_VERSION_1_2
     JavaVMInitArgs vm_args;
     JavaVMOption options[1];
     options[0].optionString =
         "-Djava.class.path=" USER_CLASSPATH;
     vm_args.version = 0x00010002;
     vm_args.options = options;
     vm_args.nOptions = 1;
     vm_args.ignoreUnrecognized = JNI_TRUE;
     /* Create the Java VM */
     res = JNI_CreateJavaVM(&jvm, (void**)&env, &vm_args);
 #else
     JDK1_1InitArgs vm_args;
     char classpath[1024];
     vm_args.version = 0x00010001;
     JNI_GetDefaultJavaVMInitArgs(&vm_args);
     /* Append USER_CLASSPATH to the default system class path */
     sprintf(classpath, "%s%c%s",
             vm_args.classpath, PATH_SEPARATOR, USER_CLASSPATH);
     vm_args.classpath = classpath;
     /* Create the Java VM */
     res = JNI_CreateJavaVM(&jvm, &env, &vm_args);
 #endif /* JNI_VERSION_1_2 */

     if (res < 0) {
         fprintf(stderr, "Can't create Java VM\n");
         exit(1);
     }
     cls = (*env)->FindClass(env, "Prog");
     if (cls == NULL) {
         goto destroy;
     }

     mid = (*env)->GetStaticMethodID(env, cls, "main",
                                     "([Ljava/lang/String;)V");
     if (mid == NULL) {
         goto destroy;
     }
     jstr = (*env)->NewStringUTF(env, " from C!");
     if (jstr == NULL) {
         goto destroy;
     }
     stringClass = (*env)->FindClass(env, "java/lang/String");
     args = (*env)->NewObjectArray(env, 1, stringClass, jstr);
     if (args == NULL) {
         goto destroy;
     }
     (*env)->CallStaticVoidMethod(env, cls, mid, args);

 destroy:
     if ((*env)->ExceptionOccurred(env)) {
         (*env)->ExceptionDescribe(env);
     }
     (*jvm)->DestroyJavaVM(jvm);
 }

Я также хочу знать разницу между передачей объекта класса из Java в C и созданием Java-класса прямо в C

1 ответ

Решение

CLASSPATH должен быть определен так же, как и при запуске виртуальной машины Java. Поэтому, если вы хотите создать класс, который находится внутри пакета com.fido.android.framework.service, вы должны поместить сгенерированный исполняемый файл C в некоторый каталог, а скомпилированный класс Java - в com/fido/android/framework/service. подкаталог. Таким образом, вам не нужно изменять определение CLASSPATH

Имя пакета должно быть передано в вызове FindClass:

cls = (*env)->FindClass(env, "com/fido/android/framework/service/Prog");
Другие вопросы по тегам