การเรียกใช้วิธี java จาก c ++ ใน Android


94

ฉันพยายามเรียกเมธอด 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.UnsantediedLinkError: getJniString คุณช่วยฉันแก้ไขข้อผิดพลาดนี้ได้ไหม
Huy Tower

@AlexTran มันนานมาแล้ว แต่ตัดสินจากข้อผิดพลาดที่คุณอาจสะกดผิดหรือไม่ได้เชื่อมโยงgetJniStringวิธีการใน java หรือใน c ตรวจสอบให้แน่ใจว่าได้เชื่อมโยงรหัส c กับ java อย่างถูกต้องโดยการนำเข้าระบบ (srsly จำสิ่งนี้ไม่ได้ทั้งหมดในตอนนี้: P)
Denys S.

1
วิธีนี้เรียกเมธอด java จาก c? เป็นonCreateวิธีการของ Java อย่างโจ่งแจ้งที่เรียก C.
John

ฉันได้รับตัวถูกดำเนินการพื้นฐานของ '->' มีประเภทที่ไม่ใช่ตัวชี้ '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 ของฉัน :)
Denys S.

ฉันได้รับผลลัพธ์เดียวกันกับสิ่งที่คุณแนะนำ
Denys S.

1
มี CallVoidMethod, CallObjectMethod และอื่น ๆ ซึ่งแต่ละประเภทมีประเภทการส่งคืนที่แตกต่างกัน เนื่องจากเมธอด messageMe ของคุณคือ (Ljava / lang / String;) V คุณจึงต้องใช้ CallVoidMethod
Matthew Willis

2
โปรดทราบว่าข้อผิดพลาดที่คุณได้รับอาจบ่งชี้ว่าวิธีการดั้งเดิมของ Java (ในโค้ด Java ของคุณ) อาจไม่ใช่ประเภทการส่งคืนที่เป็นโมฆะดังนั้น GetMethodID ไม่พบ
Matthew Willis

11

โซลูชันโพสต์โดย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();
}

1
อย่าnativeวิธีการจะต้องมีความคงที่
IgorGanapolsky
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.