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;
  ...

È stato utile?

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!

Autorizzato sotto: CC-BY-SA insieme a attribuzione
Non affiliato a StackOverflow
scroll top