Perché il mio codice JNI non trova correttamente il metodo getMessage di un jthrowable?
-
03-07-2019 - |
Domanda
Sto provando ad accedere al messaggio in un jthrowable mentre gestisco un'eccezione generata quando non riesco a trovare una classe. Tuttavia, non riesco ad accedere all'ID messaggio di getMessage () sull'oggetto jthrowable e non so perché. Ho provato a cambiare la firma di getMessage in " () Ljava / lang / String " (senza il punto e virgola alla fine, ma è necessario, giusto?) senza gioia. Sono confuso da morire. Ho anche provato a sostituire getMessage con toString e che non ha funzionato. Ovviamente sto facendo qualcosa di banalmente sbagliato qui.
Ecco il codice che sto usando:
jthrowable java_exception;
jclass java_class;
jmethodID method;
java_exception = (*jEnv)->ExceptionOccurred(jEnv);
assert (java_exception != NULL);
java_class = (*jEnv)->GetObjectClass (jEnv, java_exception);
assert (java_class != NULL);
method = (*jEnv)->GetMethodID (jEnv, java_class, "getMessage", "()Ljava/lang/String;");
if (method == NULL) {
printf ("Seriously, how do I get here?!\n");
(*jEnv)->ExceptionDescribe (jEnv);
return;
}
L'output di questo codice (tra le altre cose) è simile al seguente:
Scherzi a parte, come arrivo qui ?!
Eccezione nel thread "principale" java.lang.NoClassDefFoundError: com / planet / core360 / docgen / Processor
javap -p -s java.lang.Throwable
mi dà questo:
Compilato da " Throwable.java "
public class java.lang.Throwable estende java.lang.Object implementa java.io.Serializable {
...
public java.lang.String getMessage ();
Firma: () Ljava / lang / String;
...
Soluzione
Okay, quindi sembra che il mio problema sia che GetObjectClass
non agisce come ti aspetteresti su un jrowrowable, o almeno i risultati non sono utili ai fini di ottenere metodi. Sostituendo quella parte del codice con questo funziona:
java_class = (*jEnv)->FindClass (jEnv, "java/lang/Throwable");
method = (*jEnv)->GetMethodID (jEnv, java_class, "getMessage", "()Ljava/lang/String;");
Dannatamente strano, quello. Spero che questo aiuti qualcun altro in futuro, tuttavia.
Altri suggerimenti
Ho provato il tuo approccio e ha funzionato per me. Alcune cose però: sto usando l'interfaccia C ++ (anche se ciò non dovrebbe fare la differenza) e sto usando Java 6 aggiornamento 10, edizione x64, su Ubuntu 8.04. Forse la versione e / o la piattaforma Java utilizzate faranno la differenza.
#include <cstdio>
#include <jni.h>
int
main(int argc, char** argv)
{
if (argc != 3) {
std::fprintf(stderr, "usage: %s class message\n", argv[0]);
return 1;
}
JavaVM* jvm;
void* penv;
JavaVMInitArgs args = {JNI_VERSION_1_6};
if (jint res = JNI_CreateJavaVM(&jvm, &penv, &args)) {
std::fprintf(stderr, "Can's create JVM: %d\n", res);
return -res;
}
JNIEnv* env(static_cast<JNIEnv*>(penv));
jint vers(env->GetVersion());
std::printf("JNI version %d.%d\n", vers >> 16, vers & 0xffff);
env->ThrowNew(env->FindClass(argv[1]), argv[2]);
jthrowable exc(env->ExceptionOccurred());
std::printf("Exception: %p\n", exc);
if (exc) {
jclass exccls(env->GetObjectClass(exc));
jclass clscls(env->FindClass("java/lang/Class"));
jmethodID getName(env->GetMethodID(clscls, "getName", "()Ljava/lang/String;"));
jstring name(static_cast<jstring>(env->CallObjectMethod(exccls, getName)));
char const* utfName(env->GetStringUTFChars(name, 0));
jmethodID getMessage(env->GetMethodID(exccls, "getMessage", "()Ljava/lang/String;"));
jstring message(static_cast<jstring>(env->CallObjectMethod(exc, getMessage)));
char const* utfMessage(env->GetStringUTFChars(message, 0));
std::printf("Exception: %s: %s\n", utfName, utfMessage);
env->ReleaseStringUTFChars(message, utfMessage);
env->ReleaseStringUTFChars(name, utfName);
}
return -jvm->DestroyJavaVM();
}
Ho usato jnitest java / lang / InternalError 'Hello, world!'
per i miei test; sentiti libero di provare con diversi tipi di eccezione!