Создать класс 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");