在安卓系统中从C++调用Java方法

97

我正在尝试从C++获取一个简单的Java方法调用,而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到native的getJniString*方法调用过程中,从本地代码调用messageMe方法。

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,虽然很久以前了,但从错误来看,你可能拼错了或者没有在Java或C中链接getJniString方法。请确保通过系统导入正确地将C代码链接到Java(真的不记得这些东西了:P)。 - Denys S.
1
这是如何从C语言调用Java方法?这显然是Java的onCreate方法在调用您的本地C语言方法。 - John
当我使用环境变量(env)执行时,出现了基本操作数错误:“->”的操作数类型为“JNIEnv *”,如果想在没有env变量的情况下从JNI回调到Java层怎么办?有什么建议! - CoDe
您IP地址为143.198.54.68,由于运营成本限制,当前对于免费用户的使用频率限制为每个IP每72小时10次对话,如需解除限制,请点击左下角设置图标按钮(手机用户先点击左上角菜单按钮)。 - BartoszKP
2个回答

48

如果它是一个对象的方法,你需要将对象传递给 CallObjectMethod

jobject result = env->CallObjectMethod(obj, messageMe, jstr);
你之前做的相当于是调用了 jstr.messageMe() 方法。
由于你的方法是 void 类型,所以应该调用:
env->CallVoidMethod(obj, messageMe, jstr);
如果你想要返回结果,你需要更改 JNI 签名(括号内的 ()V 表示返回类型为 void),同时也需要在 Java 代码中更改返回值类型。

请指导我如何做到这一点,因为我的P.S. :) - Denys S.
我用你建议的方法得到了相同的结果。 - Denys S.
1
实际上有CallVoidMethod、CallObjectMethod等不同返回类型的方法。由于你的messageMe方法是(Ljava/lang/String;)V,所以你需要使用CallVoidMethod。 - Matthew
2
请注意,您收到的错误可能表明您的Java本地方法(在您的Java代码中)可能不是void返回类型,因此无法通过GetMethodID找到它。 - Matthew

13

以下是问题帖中由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

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接