Wywołanie metody Java z języka C ++ w systemie Android

92

Próbuję uzyskać proste wywołanie metody Java z C ++, podczas gdy Java wywołuje metodę natywną. Oto kod 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();
}

Próbuję wywołać messageMemetodę z kodu natywnego w procesie getJniString*wywołania metody z języka Java do natywnego.

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!");
}

Po czystej kompilacji aplikacja zatrzymuje się i wyświetla następną wiadomość:

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)

Najwyraźniej oznacza to, że nazwa metody jest nieprawidłowa, ale dla mnie wygląda dobrze.

Denys S.
źródło
21
Opublikuj swoje rozwiązanie jako zwykłą odpowiedź, aby Twoje pytanie i rozwiązanie były bardziej czytelne, a przez to bardziej przydatne dla społeczności. Możesz także współpracować z innymi osobami, które już odpowiedziały, aby uzupełnić ich odpowiedzi.
misiu_mp
@Denys: Śledziłem twoje kodowanie, ale pojawia się ten błąd: java.lang.UnsatisfiedLinkError: getJniString. Czy możesz mi pomóc naprawić ten błąd?
Huy Tower
@AlexTran, to było dawno temu, ale sądząc po błędzie prawdopodobnie błędnie wpisałeś lub nie połączyłeś getJniStringmetody ani w Javie, ani w C. Upewnij się, że poprawnie łączysz kod c z Javą prawdopodobnie przez import systemu (srsly nie pamiętam teraz tego wszystkiego: P)
Denys S.
1
Jak to jest wywołanie metody Java z c? Jest to rażąco onCreatemetoda Java dzwoniąca do twojego rodzimego C.
John
Otrzymuję bazowy operand „->” nie będący wskaźnikiem typu „JNIEnv, gdy jest wykonywany ze zmienną środowiskową (env). A co by było, gdybyśmy chcieli obejść się bez zmiennej env *, takiej jak wywołanie zwrotne z warstwy JNI do warstwy Java! Jakieś sugestie!
CoDe

Odpowiedzi:

45

Jeśli jest to metoda obiektowa, musisz przekazać obiekt do CallObjectMethod:

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

To, co robiłeś, było odpowiednikiem jstr.messageMe().

Ponieważ twoja jest metodą void, powinieneś wywołać:

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

Jeśli chcesz zwrócić wynik, musisz zmienić swój podpis JNI ( ()Voznacza metodę voidtypu zwracanego), a także typ zwracany w kodzie Java.

Matthew Willis
źródło
Proszę, poprowadź mnie, jak to zrobić, ze względu na moje PS :)
Denys S.
Taki sam rezultat otrzymuję dzięki temu, co sugerujesz.
Denys S.
1
w rzeczywistości istnieje CallVoidMethod, CallObjectMethod itp., z których każdy ma inny typ zwracania. Ponieważ twoją metodą messageMe jest (Ljava / lang / String;) V, musisz użyć CallVoidMethod.
Matthew Willis
2
zwróć uwagę, że błąd, który otrzymujesz, prawdopodobnie wskazuje, że Twoja natywna metoda Java (w kodzie Java) prawdopodobnie nie jest typu zwracanego void i dlatego nie jest odnajdywana przez GetMethodID
Matthew Willis
10

Rozwiązanie zamieszczone przez Denysa S. w pytaniu:

Całkiem zepsułem to z konwersją c do c ++ (w zasadzie envzmienne rzeczy), ale udało mi się to działać z następującym kodem dla 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);
}

I następny kod dla metod 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();
}
BartoszKP
źródło
1
Czy nativemetody muszą być statyczne?
IgorGanapolsky