Виклик методу Java з c ++ в Android


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

Я намагаюся викликати messageMeметод із власного коду в процесі getJniString*виклику методу з Java на рідний.

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. Чи можете ви допомогти мені виправити цю помилку?
Huy Tower

@AlexTran, це було дуже давно, але, судячи з помилки, ви, мабуть, неправильно написали або не зв'язали getJniStringметод ні в Java, ні в c. Переконайтеся, що правильно зв’язали код c з java, ймовірно, за допомогою імпорту системи (srsly не пам’ятає все це зараз: P)
Денис С.

1
Як це викликає метод Java із c? Це відверто onCreateметод Java, який викликає вашу рідну C.
Джон

Я отримую базовий операнд '->' має тип покажчика 'JNIEnv під час виконання із змінною середовища (env). А також, якби хотіли обійтися без змінної env *, як зворотний виклик з JNI на рівень Java! Будь-яка пропозиція!
CoDe

Відповіді:


45

Якщо це об’єктний метод, вам потрібно передати об’єкт CallObjectMethod:

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

Те, що ви робили, було еквівалентом jstr.messageMe().

Оскільки ваш метод є порожнім, вам слід викликати:

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

Якщо ви хочете повернути результат, вам потрібно змінити свій підпис JNI ( ()Vозначає спосіб voidповернення типу), а також тип повернення у вашому коді Java.


Будь ласка, дайте мені вказівки щодо того, як це зробити, через мій PS :)
Денис С.

Я отримую той самий результат із тим, що ви пропонуєте.
Денис С.

1
насправді є CallVoidMethod, CallObjectMethod тощо, кожен із різним типом повернення. Оскільки ваш метод messageMe (Ljava / lang / String;) V, вам потрібно використовувати CallVoidMethod.
Метью Вілліс,

2
зауважте, що помилка, яку ви отримуєте, імовірно, вказує на те, що ваш власний метод Java (у вашому коді Java), мабуть, не має типу void return і тому не знайдений GetMethodID
Matthew Willis

10

Рішення, опубліковане Денисом С. у дописі:

Я цілком переплутав це з перетворенням в 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методи повинні бути статичними?
Ігор Ганапольський
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.