7

クラスが見つからない場合に生成される例外を処理しながら、jthrowableでメッセージにアクセスしようとしています。しかし、jthrowableオブジェクトのgetMessage()のメッセージIDにアクセスできず、その理由がわかりません。getMessageの署名を「()Ljava / lang / String」に変更してみました(最後にセミコロンはありませんが、それは必要ですよね?)。私はこれについて地獄と混同しています。getMessageをtoStringに置き換えようとましたが、うまくいきませんでした。明らかに、私はここで些細なことをしている。

これが私が使用しているコードです:

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

このコードの出力は(とりわけ)次のようになります。

真剣に、どうやってここに着くの?
スレッド「main」の例外java.lang.NoClassDefFoundError:com / planet / core360 / docgen / Processor

javap -p -s java.lang.Throwable私にこれを与えます:

「Throwable.java」からコンパイルされた
publicclassjava.lang.Throwable extends java.lang.Objectimplements java.io.Serializable {
...
public java.lang.String getMessage();
署名:()Ljava / lang / String;
..。

4

2 に答える 2

8

さて、私の問題はGetObjectClass、jthrowableで期待どおりに動作しないか、少なくともその結果がメソッドを取得する目的に役立たないことだったようです。コードのその部分を次のように置き換えると、次のように機能します。

java_class = (*jEnv)->FindClass (jEnv, "java/lang/Throwable");
method = (*jEnv)->GetMethodID (jEnv, java_class, "getMessage", "()Ljava/lang/String;");

ひどく奇妙な、それ。ただし、これが将来他の誰かに役立つことを願っています。

于 2008-10-18T22:33:06.590 に答える
3

私はあなたのアプローチを試しましたが、うまくいきました。ただし、いくつかの点があります。私は C++ インターフェイスを使用しており (違いはありません)、Ubuntu 8.04 で Java 6 update 10、x64 エディションを使用しています。おそらく、使用する Java のバージョンやプラットフォームによって違いが生じるでしょう。

#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();
}

jnitest java/lang/InternalError 'Hello, world!'テストに使用しました。さまざまな例外タイプを試してみてください!

于 2008-10-18T22:54:34.800 に答える