AndroidでC ++からJavaメソッドを呼び出す


91

Javaがネイティブメソッドを呼び出す間に、C ++から単純なJavaメソッド呼び出しを取得しようとしています。Javaコードは次のとおりです。

public class MainActivity extends Activity {
    private static String LIB_NAME = "name";

    static {
        System.loadLibrary(LIB_NAME);
    }

    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        TextView tv = (TextView) findViewById(R.id.textview);
        tv.setText(this.getJniString());
    }

    public void messageMe(String text) {
        System.out.println(text);
    }

    public native String getJniString();
}

Javaからネイティブへのメソッド呼び出しmessageMeの過程で、ネイティブコードからメソッドを呼び出そうとしていますgetJniString*

native.cpp:

#include <string.h>
#include <stdio.h>
#include <jni.h>

jstring Java_the_package_MainActivity_getJniString( JNIEnv* env, jobject obj, jint depth ){

//    JavaVM *vm;
//    JNIEnv *env;
//    JavaVMInitArgs vm_args;
//    vm_args.version = JNI_VERSION_1_2;
//    vm_args.nOptions = 0;
//    vm_args.ignoreUnrecognized = 1;
//
//    // Construct a VM
//    jint res = JNI_CreateJavaVM(&vm, (void **)&env, &vm_args);

    // Construct a String
    jstring jstr = env->NewStringUTF("This string comes from JNI");
    // First get the class that contains the method you need to call
    jclass clazz = env->FindClass("the/package/MainActivity");
    // Get the method that you want to call
    jmethodID messageMe = env->GetMethodID(clazz, "messageMe", "(Ljava/lang/String;)V");
    // Call the method on the object
    jobject result = env->CallObjectMethod(jstr, messageMe);
    // Get a C-style string
    const char* str = env->GetStringUTFChars((jstring) result, NULL);
    printf("%s\n", str);
        // Clean up
    env->ReleaseStringUTFChars(jstr, str);

//    // Shutdown the VM.
//    vm->DestroyJavaVM();

    return env->NewStringUTF("Hello from JNI!");
}

クリーンコンパイル後、アプリは次のメッセージで停止します。

ERROR/AndroidRuntime(742): FATAL EXCEPTION: main
        java.lang.NoSuchMethodError: messageMe
        at *.android.t3d.MainActivity.getJniString(Native Method)
        at *.android.t3d.MainActivity.onCreate(MainActivity.java:22)

どうやらそれはメソッド名が間違っていることを意味しますが、私には問題ないように見えます。


21
質問とソリューションの両方を読みやすくし、コミュニティにとってより役立つように、ソリューションを通常の回答として投稿してください。また、既に回答した他のユーザーと共同して回答を完了することもできます。
misiu_mp

@Denys:私はあなたのコーディングに従いましたが、次のエラーが発生しました:java.lang.UnsatisfiedLinkError:getJniString。このエラーの修正を手伝ってもらえますか?
ホイタワー

@AlexTran、それはずっと前のことですが、エラーから判断すると、スペルが間違っているgetJniStringか、JavaまたはCでメソッドをリンクしていません。おそらくシステムのインポートによってCコードをJavaに適切にリンクするようにしてください(今のところ、これらすべてを覚えていません:P)
Denys S.

1
これはCからJavaメソッドをどのように呼び出していますか?それは露骨にJavaのだonCreate、あなたのネイティブCを呼び出す方法
ジョン

「->」のベースオペランドを取得していますが、environment(env)変数で実行すると、ポインタ型ではない「JNIEnv」が発生します。また、JNIからJavaレイヤーへのコールバックのように、env *変数なしで実行したい場合はどうでしょうか。なにか提案を!
CoDe、2015

回答:


45

オブジェクトメソッドの場合は、オブジェクトをに渡す必要がありますCallObjectMethod

jobject result = env->CallObjectMethod(obj, messageMe, jstr);

あなたがやっていたことはと同等でしたjstr.messageMe()

あなたはvoidメソッドなので、次を呼び出す必要があります:

env->CallVoidMethod(obj, messageMe, jstr);

結果を返したい場合は、JNIシグネチャ(戻り値の型の()Vメソッドを意味するvoid)とJavaコードの戻り値の型を変更する必要があります。


私のPSのために、それを行う方法を私に案内してください:)
Denys S.

私はあなたが提案しているものと同じ結果を得ます。
Denys S.

1
実際には、CallVoidMethod、CallObjectMethodなどがあり、それぞれ戻り値の型が異なります。messageMeメソッドは(Ljava / lang / String;)Vであるため、CallVoidMethodを使用する必要があります。
マシューウィリス

2
表示されているエラーは、おそらくJavaネイティブメソッド(Javaコード内)が戻り値の型がvoidではないため、GetMethodIDで見つからないことを示していることに注意してください
Matthew Willis

10

質問の投稿でDenys S.が投稿した解決策:

私はcからc ++への変換(基本的にenv変数のもの)でそれをめちゃくちゃにしましたが、C ++の次のコードで動作しました。

#include <string.h>
#include <stdio.h>
#include <jni.h>

jstring Java_the_package_MainActivity_getJniString( JNIEnv* env, jobject obj){

    jstring jstr = (*env)->NewStringUTF(env, "This comes from jni.");
    jclass clazz = (*env)->FindClass(env, "com/inceptix/android/t3d/MainActivity");
    jmethodID messageMe = (*env)->GetMethodID(env, clazz, "messageMe", "(Ljava/lang/String;)Ljava/lang/String;");
    jobject result = (*env)->CallObjectMethod(env, obj, messageMe, jstr);

    const char* str = (*env)->GetStringUTFChars(env,(jstring) result, NULL); // should be released but what a heck, it's a tutorial :)
    printf("%s\n", str);

    return (*env)->NewStringUTF(env, str);
}

そして、Javaメソッドの次のコード:

    public class MainActivity extends Activity {
    private static String LIB_NAME = "thelib";

    static {
        System.loadLibrary(LIB_NAME);
    }

    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        TextView tv = (TextView) findViewById(R.id.textview);
        tv.setText(this.getJniString());
    }

    // please, let me live even though I used this dark programming technique
    public String messageMe(String text) {
        System.out.println(text);
        return text;
    }

    public native String getJniString();
}

くださいnative方法は静的である必要がありますか?
IgorGanapolsky
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.